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Access Tutorial 1: Introduction to Microsoft Access 


The purpose of these tutorials is not to teach you 
Microsoft Access, but rather to teach you some 
generic information systems concepts and skills 
using Access. Of course, as a side effect, you will 
learn a great deal about the software—enough to 
write your own useful applications. However, keep in 
mind that Access is an enormously complex, nearly- 
industrial-strength software development environ¬ 
ment. The material here only scrapes the surface of 
Access development and database programming. 

1.1 Introduction: What is Access? 

Microsoft Access is a relational database manage¬ 
ment system (DBMS). At the most basic level, a 
DBMS is a program that facilitates the storage and 
retrieval of structured information on a computer’s 
hard drive. Examples of well-know industrial-strength 
relational DBMSes include 
• Oracle 
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Last update: 24-Aug-1997 


1. Introduction to Microsoft Access 

• a full-featured procedural programming lan¬ 
guage —essentially a subset of Visual Basic, 

• a simplified procedural macro language unique 
to Access; 

• a rapid application development environment 

complete with visual form and report develop¬ 
ment tools; 

• a sprinkling of objected-oriented extensions; 

and, 

• various wizards and builders to make develop¬ 
ment easier. 

For new users, these “multiple personalities” can be 
a source of enormous frustration. The problem is 
that each personality is based on a different set of 
assumptions and a different view of computing. For 
instance, 

• the relational database personality expects you 
to view your application as sets of data; 


• Microsoft SQL Server 

• IBM DB2 

• Informix 

Well-know PC-based (“desktop”) relational DBMSes 
include 

• Microsoft Access 

• Microsoft FoxPro 

• Borland dBase 

1.1.1 The many faces of Access 

Microsoft generally likes to incorporate as many fea¬ 
tures as possible into its products. For example, the 
Access package contains the following elements: 

• a relational database system that supports two 
industry standard query languages: Structured 
Query Language (SQL) and Query By Example 
(QBE); 
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Introduction: What is Access? 

• the procedural programming personality expects 
you to view your application as commands to be 
executed sequentially; 

• the object-oriented personality expects you to 
view your application as objects which encapsu¬ 
late state and behavior information. 

Microsoft makes no effort to provide an overall logi¬ 
cal integration of these personalities (indeed, it is 
unlikely that such an integration is possible). Instead, 
it is up to you as a developer to pick and choose the 
best approach to implementing your application. 

Since there are often several vastly different ways to 
implement a particular feature in Access, recogniz¬ 
ing the different personalities and exploiting the best 
features (and avoiding the pitfalls) of each are impor¬ 
tant skills for Access developers. 

The advantage of these multiple personalities is that 
it is possible to use Access to learn about an enor¬ 
mous range of information systems concepts without 
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1. Introduction to Microsoft Access 


Learning objectives 


having to interact with a large number of “single-per¬ 
sonality” tools, for example: 

• Oracle for relational databases 

• PowerBuilder for rapid applications development, 

• SmallTalk for object-oriented programming. 

Keep this advantage in mind as we switch back and 
forth between personalities and different computing 
paradigms. 

1.1.2 What is in an Access database 
file? 

Although the term “database” typically refers to a col¬ 
lection of related data tables, an Access database 
includes more than just data. In addition to tables, an 
Access database file contains several different types 
of database objects: 

• saved queries for organizing data, 

• forms for interacting with the data on screen, 

• reports for printing results, 


• macros and Visual Basic programs for extending 
the functionality of database applications. 

All these database objects are stored in a single file 
named <f iiename> .mdb. When you are running 
Access, a temporary “locking” file named <fiie- 
name>. ldb is also created. You can safely ignore 
the Mdb file; everything of value is in the *.mdb file. 

1.2 Learning objectives 

□ How do I get started? 

□ How do I determine the version I am using? 

□ How do I create or edit a database object? 

□ What is the database window and what does 
it contain? 

□ How do I import an Excel spreadsheet? 

□ How do I delete or rename database objects? 
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1. Introduction to Microsoft Access 

□ How do I get help from the on-line help 
system? 

□ How do I compact a database to save space? 

1.3 Tutorial exercises 

In this tutorial, you will start by creating a new data¬ 
base file. 

1.3.1 Starting Access 

• To start Access, you double click the Access icon 
(||& for version 8.0 and 7.0 or for version 
2.0) from within Microsoft Windows. 

If you are working in the Commerce PC Lab, you will 
be working with Access version 2.0. If you are work¬ 
ing at home, you will able be to tell what version you 
are using by watching the screen “splash” as the pro¬ 
gram loads. Alternatively, select Help > About 


Tutorial exercises 

Access from the main menu to see which version 
you are using. 

A All the screen shots in these tutorials are 
. taken from Access version 7.0 (released as 
part of Office 95). Although there are some 
important differences between version 2.0 
and version 7.0, the concepts covered here 
are the same for both. Version 8.0 (released 
as part of Office 97) is only slightly different 
from version 7.0. 

A Whenever the instructions given in the tutorial 
. differ significantly from version 7.0, a warning 
box such as this is used. 

1.3.2 Creating a new database 

• Follow the directions in Figure 1.1 to create a 
new database file called myfiie.mdb. 
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1. Introduction to Microsoft Access 


Tutorial exercises 


FIGURE 1.1: Select the name and location of your new (empty) database. 
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version 2.0. 
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• Examine the main features of the database win¬ 
dow—including the tabs for viewing the different 
database objects—as shown in Figure 1.2. 

1.3.3 Opening an existing database 

Since an empty database file is not particularly inter¬ 
esting, you are provided with an existing database 
file containing information about university courses. 
For the remainder of this tutorial, we will use a file 
called univ0_v7 .mdb, which is available from the 
tutorial’s Internet site. 

A If you are using version 2.0, you will need to 
. use the univ0_v2 .mdb database instead. 
Although you can open a version 2.0 data¬ 
base with version 7.0, you cannot open a ver¬ 
sion 7.0 database with version 2.0. Importing 
and exporting across versions is possible, 
however. 


Tutorial exercises 

A If you are using version 8.0, you can use 

. either univ0_v2 .mdb Or univ0_v7 .mdb for 

the tutorials. When you open the file, Access 
will ask you if you want to convert it to version 
8.0. Select yes and provide a new name for 
the converted file (e.g., univ0_v8 .mdb) 

• Open the univ0_vx.mdb file and examine the 
contents of the Sections table, as shown in 
Figure 1.3. 

1.3.4 Importing data from other 
applications 

Access makes it easy to import data from other 
applications. In this section, you will create a new 
table using data from an Excel spreadsheet. 

• Select File > Get External Data > Import from the 
main menu and import the depts. xis spread- 
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1. Introduction to Microsoft Access 


Tutorial exercises 


FIGURE 1.2: The database window contains all the database objects tor a particular application, 


Tables — 
contain data 
in rows and 
columns. 


'% Microsoft Access 


File Edit View insert Tools Window^Help 


© 


The database window is always 
available from the Window menu. 


Queries — allow the 
information in 
tables to be sorted, 
filtered, and shown 
in different ways. 



Forms — are for 
displaying 
information on 
the screen. 


Reports —are 
for organizing 
and printing 
information. 


Modules — 
contain Visual 
Basic 

procedures and 
functions. 


Macros — are sets of high- 
level commands that can be 
used to process data and 
perform repetitive tasks. 
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Tutorial exercises 


FIGURE 1.3: Open the univO_vx .mdb file for the version of Access that you are using and then 

open the Sections table 
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1. Introduction to Microsoft Access 


Tutorial exercises 


sheet as a new table called Departments (see 
Figure 1 .4). 

A in version 2.0, the menu structure is slightly 
. different. As such, you must use File > Import. 

• Use the import wizard specify the basic import 
parameters. You should accept all the defaults 
provided by the wizard except for those shown in 
Figure 1.5. 

• Double click the Departments table to ensure it 
was imported correctly. 



If you make a mistake, you can rename or 
delete a table (or any database object in the 
database window) by selecting it and right- 
clicking (pressing the right mouse button 
once). 


1.3.5 Getting help 

A recent trend in commercial software (especially 
from Microsoft) is a reliance on on-line help and doc¬ 
umentation in lieu of printed manuals. As a conse¬ 
quence, a good understanding of how to use the on¬ 
line help system is essential for learning any new 
software. In this section, you will use Access’ on-line 
help system to tell you how to compact a database. 

• Press FI to invoke the on-line help system. Find 
information on compacting a database, as shown 
in Figure 1.6. 

• Familiarize yourself with the basic elements of 
the help window as shown in Figure 1.7. 

1.3.6 Compacting your database 

• Follow the directions provided by the on-line help 
window shown in Figure 1.7 to compact your 
database. 
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FIGURE 1.4: Import the dept. xis spreadsheet as a table called Departments. 
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Tutorial exercises 


FIGURE 1.5: Use the spreadsheet import wizard to import the Excel file. 
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Help Topics: Microsoft Access for Windows 95 
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Troubleshootcomp acti n g d atab as e s 


© 


The Index is the best place to 
start when you are looking for a 
specific topic. If you need more 
structured information or are 
looking for an overview, use the 
Contents tab. 


© 


Display 


Cancel 


—■ - 1 - 


Double click the most 
promising entry in this list 
to get the actual help topic. 


For most students, the help 
system in Access version 
2.0 is easier to navigate. 
Use the “cue cards” in 
version 2.0 to get step-by- 
step instructions for many 
operations. 
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1. Introduction to Microsoft Access 


Tutorial exercises 


FIGURE 1.7: Follow the instructions provided by help to compact your database 


Microsoft Access for Windows 95 


Help Topics 


Options 


Compact aftafcahase to defragment the 
file and free disk sp; 

delete ta b I d ata base can be col 

iSjSgiSij e nte d efficie ntip 

Compacting the database makes a copy of the 
database, rearrangihli^S^lfe^tabase file is stojfjBijf; 

iiiii^lose the database. If you a 
i;:® ll us 

iiiiM ata I a lljiijiif 

and then click Compact Databa sSjiiiijijijiilijjgjjig: 
:3:>|S|he Databa dialogi^S^:: 
sp^lliiii^&tabase you wanjtitS:iSdi^|S^i^ 

Into dialog box, specify 
l d e nj^f jliiiSgiiiiSSiTn p a cte d 

database. 


Minimize (rather than close) help 
when you are working so that you can 
use the Back button to return to 
previously visited topics without 
repeating the search. 


Press help topics to return to the 
index. 


Words underlined with a dashed line 
provide important definitions. 


1. Introduction to Microsoft Access 

1.4 Discussion 

1.4.1 The database file in Access 

The term “database” means different things depend¬ 
ing on the DBMS used. For example in dBase IV, a 
database is a file (<fiiename>. dbf) containing a 
single table. Forms and reports are also stored as 
individual files with different extensions. The net 
result is a clutter of files. 

In contrast, an Oracle database has virtually no rela¬ 
tionship to individual files or individual projects. For 
instance, a database may contain many tables from 
different projects/applications and may also be 
stored split into one or more files (perhaps on differ¬ 
ent machines). 

Access strikes a convenient balance—all the 
“objects” (tables, queries, forms, reports, etc.) for a 
single project/application are stored in a single file. 
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Discussion 

1.4.2 Compacting a database 

As the help system points out, Access database files 
can become highly fragmented and grow to become 
much larger than you might expect given the amount 
of data they contain (e.g., multiple megabytes for a 
handful of records). Compacting the database from 
time to time eliminates fragmentation and can dra¬ 
matically reduce the disk space requirement of your 
database. 

1.4.3 Renaming a database 

It is often the case that you are working with a data¬ 
base and want to save it under a different name or 
save it on to a different disk drive. However, one 
command on the File menu that is conspicuous by its 
absence is Save As. 

However, when compacting your database, Access 
asks for the name and destination of the compacted 
file. As a result, the compact database utility can be 
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1. Introduction to Microsoft Access 


Discussion 


used as a substitute for the Save As command. This 
is especially useful in situations in which you cannot 
use the operating system to rename a file (e.g., 
when you do not have access to the Windows file 
manager). 

1.4.4 Developing applications in Access 

In general, there are two basic approaches to devel¬ 
oping information systems: 

• in-depth systems analysis, design, and imple¬ 
mentation, 

• rapid prototyping (in which analysis, design, and 
implementation are done iteratively) 

Access provides a number of features (such as 
graphical design tools, wizards, and a high-level 
macro language) that facilitate rapid prototyping. 
Since you are going to build a small system and 
since time is limited, you will use a rapid prototyping 
approach to build your application. The recom¬ 


mended sequence for prototyping using Access is 

the following: 

1. Model the information of interest in terms of enti¬ 
ties and relationships between the entities (this is 
covered in the lecture portion of the course). 

2. Create a table for each entity (Tutorial 2). 

3. Specify the relationships between the tables 
(Tutorial 3). 

4. Organize the information in your tables using 
queries (Tutorial 4, Tutorial 5, Tutorial 10) 

5. Create forms and reports to support input and 
output transactions (Tutorial 6, Tutorial 7). 

6. Enhance you forms with input controls 
(Tutorial 8) 

7. Create action queries (Tutorial 11), macros 
(Tutorial 13), or Visual Basic programs 
(Tutorial 12, Tutorial 14) to perform the transac¬ 
tion processing functions of the application. 
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1. Introduction to Microsoft Access 

8. Create “triggers” (procedures attached to events) 
to automate certain repetitive tasks (Tutorial 15). 

1.4.5 Use of linked tables 

Most professional Access developers do not put their 
tables in the same database file as their queries, 
forms, reports, and so on. The reason for this is sim¬ 
ple: keep the application’s data and interface sepa¬ 
rate. 

Access allows you to use the “linked table” feature to 
link two database files: one containing all the tables 
(“data”) and another containing all the interface and 
logic elements of the application (“interface”). The 
linked tables from the data file show up in the inter¬ 
face file with little arrows (indicating that they are not 
actually stored in the interface file). 

In this way, you can modify or update the interface 
file without affecting the actual data in any way. You 
just copy the new interface file over to the user’s 


Application to the assignment 

machine, update the links to the data file, and the 
upgrade is done. 

A Do not used linked tables in the assignment. 

. The links are dependent on the absolute 
directory structure. As a result, if the directory 
structure on your machine is different from 
that on the marker’s machine, the marker will 
not be able to use your application without 
first updating the links (a time consuming pro¬ 
cess for a large number of assignments). 

1.5 Application to the assignment 

After completing this tutorial you should be ready to 
create the database file that you will use for the 
remainder of the course. 

1. Create an empty database file called <your 
groupiD>. mdb. Remember that your group 
number consists of eight digits. 
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1. Introduction to Microsoft Access 


Application to the assignment 


2. Import the inventor. xis spreadsheet as your 
Products table. 

3. Use the compact utility to make a backup copy of 
your database (use a different name such as 

backup. mdb). 
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Access Tutorial 2: Tables 


2.1 Introduction: The importance 
of good table design 

Tables are where data in a database is stored; con¬ 
sequently, tables form the core of any database 
application. In addition to basic data, Access permits 
a large amount of domain knowledge (such as cap¬ 
tions, default values, constraints, etc.) to be stored at 
the table level. 

A Extra time spent thinking about table design 
. can result in enormous time savings during 
later stages of the project. Non-trivial changes 
to tables and relationships become increas¬ 
ingly difficult as the application grows in size 
and complexity. 


© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 25-Aug-1997 

2. Tables 

2.3.1 Datasheet basics 

• If you have not already done so, open the 
univO_vx.mdb database file from Tutorial 1 . 

• Open the Departments table. The important 
elements of the datasheet view are shown in 
Figure 2.1 . 

• Use the field selectors to adjust the width of the 
DeptName field as shown in Figure 2.1. 

• Add the Biology department (BIOL) to the table, 
as shown in Figure 2.2. 

• Delete the “Basket Weaving” record by clicking 
on its record selector and pressing the Delete 
key. 

2.3.2 Creating a new table 

In this section you will create and save a very basic 
skeleton for table called Employees. This table 
could be used to keep track of university employees 


2.2 Learning objectives 

□ How do I enter and edit data in the datasheet 
view of a table? 

□ How do I create a new table? 

□ How do I set the primary key for a table? 

□ How do I specify field properties such as the 
input mask and caption? 

□ Why won’t an autonumber field restart 
counting at one? 

□ What are the different types of keys? 

2.3 Tutorial exercises 

In this tutorial, you will learn to interact with existing 
tables and design new tables. 
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Tutorial exercises 

such as lecturers, department heads, departmental 
secretaries, and so on. 

• Return to the database window and create a new 
table as shown in Figure 2.3. 

• In the table design window shown in Figure 2.4, 
type in the following information: 


Field name 

Data type 

Description 

(optional) 

EmployeelD 

Text 

use employee 

S.I.N. 

FName 

Text 

First name 

LName 

Text 

Last name 

Phone 

Text 


Salary 

Currency 



• Select File > Save from the main menu (or press 
Controls) and save the table under the name 

Employees. 
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2. Tables 


Tutorial exercises 


FIGURE 2.1: The datasheet view of the Departments table. 


The field names are shown in the “field 
selectors” across the top of the columns. 


© 


You can temporarily sort the records 
in a particular order by right-clicking 
any of the field selectors. 



The records are shown as rows. 


The black triangle indicates the 
current record”. 


a 


The grey boxes are “record selectors”. 


Resize the Dept Name column by clicking near 
the column border and dragging the border to 
the right. 



Departments : Table 

W 


— i 

DeptCode 

Depth ame 

r Huilding 


► 

if 

istaam 

B a s ket We a vi nfMKIG U 


r 

COMM 

Commerce and 

ANGU 



CRVVR 

Creative Writing 

BLJCH 


j 

EDUC 

Education 

SCRF 


V 

ENGL 

English 

BLJCH 



MATH 

Math 

MATH 



MUSC 

Music 

The asterisk (*) indicates a 

¥ 



place holder for a new record. 


The “navigation buttons” at the bottom of the window 
indicate the current record number and allow you to go 
directly to the first, previous, next, last, or new record. 



I h !►*! of 
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Tutorial exercises 


FIGURE 2.2: Adding and saving a record to the table. 


Add a new record by clicking in the DeptCode field 
of the “new record” field (marked by the asterisk). 


Departments : Table 



C 

eptCode DeptName Building 


BSt 

cvv 

M Departments : Table 


CGI 

vIM 


DeptCode 

DeptName 

Building 


CR^ 

NR 


BSKVV 

Basket Weaving 

ANGU 


EDI 

JC 


COMM 

Commerce and Business Administration 

ANGU 


EMC 

C 


CRVVR 

Creative Writing 

BLJCH 


MA 



EDUC 

Education 

SCRF 


MLJBC 


ENGL 

English 

BLJCH 

-w 

BIOL 


MATH 

Math 

MATH 

0 



MUSC 

Music 

MUSC 

\ 

r 

BIOL 

Biology 

BIOL 


To permanently save the change to the 
data, click on the record selector (note the 
icon changes from a pencil to a triangle). 



It is seldom necessary to 
explicitly save new 
records (or changes to 
existing records) since 
Access automatically 
saves whenever you 
move to another record, 
close the table, quit 
Access, etc. 
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2. Tables 


Tutorial exercises 


FIGURE 2.3: Create a new table. 


Click the New button to 
create a new table. 


ip univO v7 : Database 


Tables | nP Queries 

Cata 
Courses 
Departments 
Sections 


Forms | 9 Reports) S Macros| 4}- Module \ 



New Table 


Select “design view” (avoid using 
the table wizard at this point). 



Datasheet View 


Desiqn View 


Create a new table in Design 
view. 


Table Wizard 
Import Table 
LinkTable 


OK 


Cancel 
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2. Tables Tutorial exercises 

FIGURE 2.4: Use the table design window to enter the field properties for the Employees table. 


Enter the field names and 
data types for the five fields. 



The “description” column allows 
you to enter a short comment 
about the field (this information 
is not processed in any way by 
Access). 


The “field properties” section 
allows you to enter information 
about the field and constraints on 
the values for the field. 


Employees : Table 


mu 


Field Properties 


Lookup 


General 

Field Size 
Format 
Input Mask 
Caption 
Default Value 
Validation Rule 
Validation Text 
Required 
Allow Zero Length 
Indexed 



I i cates 


OK) 



Field Name 

Data Type 

Description 


■ 

f mployeelD 

T ext ^ 

use employee S.I.N. 



FNarne 

Text^^^^^ 




LNarne 

Text 




Phone 

T ext 





Currency 









Afield name 
can be up to 
El A characters 
long., 
including 
spaces. 
Press FI for 
help on field 
names. 
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2. Tables 


Tutorial exercises 


2.3.3 Specifying the primary key 

Tables normally have a primary key that uniquely 
identifies the records in the table. When you desig¬ 
nate a field as the primary key, Access will not allow 
you to enter duplicate values into the field. 

• Follow the steps in Figure 2.5 to set the primary 
key of the table to EmployeeiD. 

2.3.4 Setting field properties 

In this section, you will specify a number of field 
properties for the EmployeeiD field, as shown in 
Figure 2.6. 

• Since we are going to use the employees’ Social 
Insurance Number (S.I.N.) to uniquely identify 
them, set the Field Size property to 11 characters 
(9 for numbers and 2 for separating spaces) 

• Set the Input Mask property to the following: 

000 \ 000 \ 000;0 

• Set the Caption property to Employee id 


FIGURE 2.6: Set the field properties for the 

EmployeeiD field. 


Employees : Table 



Field Name 

Data Type 



EmployeeiD 

Text 

use employee S.I.N. 


FNarne 

Text 

first name 


LName 

T ext 

last name 


Phone 

Text 



Salary 

Currency 







General | Lookup 



Field Size 

11 


Format 


d 

Input Mask 

000\ 000\ 000:0 


Caption 

Employee ID 


Default Value 



Validation Rule 



Validation Text 



Required 

No 


Allow Zero Length 

No 


Indexed 

Yes (No Duplicates) 
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Tutorial exercises 


FIGURE 2.5: Set the primary key for the Employees table. 


Microsoft Access 


Edit 


View insert Tools Window Help 
Un d o Pro p e rty S etti n g Ctrl+Z 





— 

Delete 

— 

Delete Row 

Del 


S Employees : Table 


Field Name 



► 

EmployeeiD 

Select All 

Ctrl + A 

1 


FNarne 




LName 

Primary Key II 


Phone 

l^^jl i ' * "zl/l 

I 


Salary 


© 


Click on the grey box beside the field (or 
fields) that form the primary key. 


To select more than one field for use as the 
primary key, hold down the Control key 
while clicking on the grey boxes. 


Either click the key-shaped icon in the tool bar or 
select Edit > Primary Key from the menu. 
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2. Tables 


Discussion 


• Select View > Datasheet from the main menu to 
switch to datasheet mode as shown in Figure 2.7. 
Enter your own S.I.N. and observe the effect of 
the input mask and caption on the EmployeeiD 
field. 

• Select View > Table Design from the main menu 
to return to design mode. 

• Set the field properties for FName and LName 
(note that Length and Caption are the only two 
properties that are relevant for these two fields) 

2.3.5 Using the input mask wizard 

In this section, you will use the input mask wizard to 
create a complex input mask for a standard field 
type. You will also use the help system to learn more 
about the meaning of the symbols used to create 
input masks. 

• Select the Phone field, move the cursor to the 
input mask property, and click the button with 


three small dots (_d) to invoke the input mask 
wizard. 

• Follow the instructions provided by the wizard as 
shown in Figure 2.8. 

• Press FI while the cursor is still in the input mask 
property. Scroll down the help window to find the 
meaning of the “0”, “9”, “>” and “L” input mask 
symbols. 

2.4 Discussion 

2.4.1 Key terminology 

A key is one or more fields that uniquely determine 
the identity of the real-world object that the record is 
meant to represent. For example, there is a record in 
the student information system that contains infor¬ 
mation about you as a student. To ensure that the 
record is associated with you and only you, it con- 
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Discussion 


FIGURE 2.7: Observe the effect of the input mask and caption properties on the behavior of the 

EmployeeiD field during data entry 


Try entering various characters and 
numbers into the EmployeeiD 
field. 


If a caption is specified, it replaces the 
field name in the field selector. 

Note that the input mask will not let you 
type any characters other than numbers 
from 0-9. In addition, the spaces between 
the groups of numbers are added 
automatically. 



Press the Escape key when you are 
done to clear the changes to the record. 


Employees : Table 


ansi 



v Employee ID 

FName 

LName 

Phone 

Sal 

► 

123 456 789 





* 








Input masks provide a relatively easy way to 
avoid certain basic data input errors without 
having to write complex error checking 
programs. Note, however, that it is possible to 
over-constrain a field so that users are unable to 
enter legitimate values. 
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2. Tables 


Discussion 


FIGURE 2.8: Use the input mask wizard to create an input mask. 


© 


Select “phone 
number” from the 
list of commonly- 
used field types. 


The items in this 
list depend on the 
“international 
settings” specified 
for Windows (e.g., 
“Zip Code” may 
show instead of 
“Postal Code”). 


In Step 2, you may 
edit the input mask 
(e.g., remove the 
area code section). 


Input Mask Wizard 


Which input mask matches how you want data to look? 

To see how a selected mask works., use the Try It box. 
VTo change the Input Mask list, click the Edit List button. 


Inlut Mask: 


Data Look: 


Phone Number 


Social Insurance Num ber 
Postal Code 
Password 


206) 555-1 21 2 


EBB 333 EBB 


Input Mask Wizard 


Input Mask Wizard 


How do you want to store the data? 

With the symbols in the mask, Iike this: 
(206) BBB-1212 

r Witho ut the sym b ols i n the mas k. Ii ke this : 
206BBB1212 


M e d i u m D ate D o y o u want to ch an g e th e i n p ut mi as k? 
Short Date 

Input Mask Name: Phone Number 


Try It: r Input Mask: 



!(999) 000-0000 


Edit List 


What placeholder character do you want 
Placeholders are replaced as you enter t 

I 3 


PI ace holder ch aracte r: 


© 


Since the input mask controls how 
the information in the field looks, it 
is possible to save some disk space 
by storing the data without the 
extras symbols, spaces, etc. For the 
size of system we are building, 
however, this savings is negligible. 


Cancel 


< Back 


Next > 
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2. Tables 

tains a field called “student number” that is guaran¬ 
teed to be unique. 

The advantage of using student number as a key 
instead of some other field—like “student name”—is 
that there may be more than one person with the 
same first and last name. The combination of stu¬ 
dent name and address is probably unique (it is 
improbable that two people with the same name will 
at the same address) but using these two fields as a 
key would be cumbersome. 

Since the terminology of keys can be confusing, the 
important terms are summarized below. 

1 . Primary key — The terms “key” and “primary 
key” are often used interchangeably. Since there 
may be more than one candidate key for an 
application, the designer has to select one: this is 
the primary key. 

2. Concatenated key: The verb “concatenate” 
means to join together in a series. A concate- 


Discussion 

nated key is made by joining together two or 
more fields. Course numbers at UBC provide a 
good example of a concatenated key made by 
joining together two fields: Deptcode and 
crsNum. For example, department alone cannot 
be the primary key since there are many courses 
in each department (e.g., COMM 335, COMM 
391). Similarly, course number cannot be used as 
a key since there are many courses with the 
same number in different departments (e.g., 
COMM 335, HIST 335, MATH 335). However, 
department and course number together form a 
concatenated key (there is only one COMM 335). 

3. Foreign key: In a one-to-many relationship, a 
foreign key is a field (or fields) in the “child” 
record that uniquely identifies the correct “parent” 
record. For example, Deptcode and CrsNum in 
the Sections table are foreign keys since these 
two keys taken together are the primary key of 
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2. Tables 


Discussion 


the Courses table. Foreign keys are identified in 
Access by creating relationships (see Tutorial 3). 

2.4.2 Fields and field properties 

2.4.2.1 Field names 

Access places relatively few restrictions on field 
names and thus it is possible to create long, descrip¬ 
tive names for your fields. The problem is that you 
have to type these field names when building que¬ 
ries, macros, and programs. As such, a balance 
should be struck between readability and ease of 
typing. You are advised to use short-but-descriptive 
field names with no spaces. 

For example, in Section 2.3.2 you created a field 
with name FName. However, you can use the caption 
property to provide a longer, more descriptive label 
such as First name. The net result is a field name 
that is easy to type when programming and a field 
caption that is easy to read when the data is viewed. 


In addition, you can use the comment field in the 
table design window to document the meaning of 
field names. 



It is strongly recommended that you avoid all 
non-alphanumeric characters whenever you 
name a field or database object. Although 
Access will permit you to use names such as 
Customer#, non-alphanumeric characters 
(such as #, /, $, %, ~, @, etc.) may cause 
undocumented problems later on. 


2.4.2.2 Data types 

The field's data type tells Access how to handle the 
information in the field. For instance, if the data type 
is date/time, then Access can perform date/time 
arithmetic on information stored in the field. If the 
same date is stored as text, however, Access treats 
it just like any other string of characters. Normally, 
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the choice of data type is straightforward. However, 

the following guidelines should be kept in mind: 

1 . Do not use a numeric data type unless you are 
going to treat the field as a number (i.e., perform 
mathematical operations on it). For instance, you 
might be tempted to store a person's student 
number as an integer. However, if the student 
number starts with a zero, then the first digit is 
dropped and you have to coerce Access into dis¬ 
playing it. Similarly, a UBC course number (e.g., 
335) might be considered a number; however, 
since courses like 439B have to accommodated, 
a numeric data type for the course number field is 
clearly inappropriate. 

2. Access provides a special data type called Auto 
Number (Counter in version 2.0). An autonum¬ 
ber/counter is really a number of type Long Inte¬ 
ger that gets incremented by Access every time 
a new record is added. As such, it is convenient 


Discussion 

for use as a primary key when no other key is 
provided or is immediately obvious. 

A Since an autonumber is really Long Integer 
. and since relationships can only be created 
between fields with the same data type, it is 
important to remember that if an autonumber 
is used on the “one” side of a relationship, a 
long integer must be used for the “many” side. 

2.4.2.3 “Disappearing” numbers in 
autonumber fields 

If, during the process of testing your application, you 
add and delete records from a table with an auto¬ 
number key, you will notice that the deleted keys are 
not reclaimed. 

For instance, if you add records to your Customer 
table (assuming that custiD is an autonumber), you 
will have a series of CustiD values: 1,2, 3... If you 
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later delete customer 1 and 2, you will notice that 
your list of customers now starts at 3. 

Clearly, it would be impossible for Access to renum¬ 
ber all the customers so the list started at 1. What 
would happen, for instance, to all the printed 
invoices with CustiD = 2 on them? Would they refer 
to the original customer 2 or the newly renumbered 
customer 2? 

A The bottom line is this: once a key is 

. assigned, it should never be reused, even if 
the entity to which it is assigned is subse¬ 
quently deleted. Thus, as far as you are con¬ 
cerned, there is no way to get your customers 
table to renumber from CustiD = 1 . 

Of course, there is a long and complicated way to do 
it, but since used an autonumber in the first place, 
you do not care about the actual value of the key— 
you just want it to be unique. In short, it makes abso¬ 


lutely no difference whether the first customer in your 
customers table is CustiD = 1 or 534. 

2.4.2.4 Input masks 

An input mask is a means of restricting what the user 
can type into the field. It provides a “template” which 
tells Access what kind of information should be in 
each space. For example, the input mask >llll 
consists of two parts: 

1. The right brace > ensures that every character 
the user types is converted into upper case. 
Thus, if the user types comm, it is automatically 
converted to comm. 

2. The characters llll are place holders for letters 
from A to Z with blank spaces not allowed. What 
this means is that the user has to type in exactly 
four letters. If she types in fewer than four or 
types a character that is not within the A to Z 
scope (e.g., &, 7, %), Access will display an error 
message. 
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There are a large number of special symbols used 
for the input mask templates. Since the meaning of 
many of the symbols is not immediately obvious, 
there is no requirement to remember the character 
codes. Instead, simply place the cursor on the input 
mask property and press FI to get help. In addition, 
the wizard can be used to provide a basic input mask 
which can later be modified. 

2.4.2.5 Input masks and literal values 

To have the input mask automatically insert a char¬ 
acter (such as a space or a dash) in a field, use a 
slash to indicate that the character following it is a lit¬ 
eral. 

For example, to create an input mask for local tele¬ 
phone numbers (e.g., 822-6109), you would use the 
following template: 000 \- 0000;0 (the dash is a lit¬ 
eral value and appears automatically as the user 
enters the telephone number). 


Discussion 

The semicolon and zero at the end of this input mask 
are important because, as the on-line help system 
points out, an input mask value actually consists of 
three parts (or “arguments”), each separated by a 
semicolon: 

• the actual template (e.g., ooo\-oooo), 

• a value (0 or 1) that tells Access how to deal with 
literal characters, and 

• the character to use as a place holder (showing 
the user how many characters to enter). 

When you use a literal character in an input mask, 
the second argument determines whether the literal 
value is simply displayed or displayed and stored in 
the table as part of the data. 

For example, if you use the input mask ooo\- 
0 0 00 ; l, Access will not store the dash with the tele¬ 
phone number. Thus, although the input mask will 
always display the number as “822-6109”, the num¬ 
ber is actually stored as “8226109”. By using the 
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2. Tables 


Application to the assignment 


input mask 000 \- 0000 ; 0 , however, you are telling 
Access to store the dash with the rest of the data. 

A If you use the wizard to create an input mask, 

. it asks you a simple question about storing lit¬ 
eral values (as shown in Figure 2.8) and fills 
in the second argument accordingly. How¬ 
ever, if you create the input mask manually, 
you should be aware that by default, Access 
does not store literal values. In other words, 
the input mask 000\-0000 is identical to the 
input mask 000\-0000;i. This has impor¬ 
tant consequences if the field in question is 
subject to referential integrity constraints (the 
value “822-6109” is not the same as 
“8226109”). 


2.5 Application to the assignment 

You now have the skills necessary to implement your 
tables. 

• Create all the tables required for the assignment. 

• Use the autonumber data type (counter in version 
2.0) for your primary keys where appropriate. 

• Specify field properties such as captions, input 
mask, and defaults where appropriate. 



If you create an input mask for ProductiD, 
ensure you understand the implications of 
Section 2.4.2.5. 


• Set the Default property of the orderDate field 
so that the current date is automatically inserted 
into the field when a new order is created (hint: 
see the Date () function in the on-line help sys¬ 
tem). 
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• Do not forget to modify your Products table (the 
data types, lengths, and field properties of 
imported tables normally need to be fine tuned) 

• Populate (enter data into) your master tables. Do 
not populate your transaction tables. 



For the purpose of the assignment, the term 
“transaction” tables refers to tables that con¬ 
tain information about individual transactions 

(e.g., Orders, OrderDetails, Ship¬ 
ments, ShipmentDetails). “Master” 

tables, in contrast, are tables that either do 
not contain information about transactions 
(e.g., Customers) or contain only summary 
or status information about transactions (e.g., 

Backorders). 


Application to the assignment 
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Access Tutorial 3: Relationships 


3.1 Introduction: The advantage of 
using tables and relationships 

A common mistake made by inexperienced data¬ 
base designers (or those who have more experience 
with spreadsheets than databases) is to ignore the 
recommendation to model the domain of interest in 
terms of entities and relationships and to put all the 
information they need into a single, large table. 
Figure 3.1 shows such a table containing information 
about courses and sections. 

• If you have not already done so, open the 
univO_vx.mdb database. 

• Open the catalog view table. 

The advantage of the single-table approach is that it 
requires less thought during the initial stages of 
application development. The disadvantages are too 
numerous to mention, but some of the most impor¬ 
tant ones are listed below: 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 22-Aug-1997 

3. Relationships 


1. Wasted space — Note that for COMM 290, the 
same basic course information is repeated for 
every section. Although the amount of disk space 
wasted in this case is trivial, this becomes an 
important issue for very large databases. 

2. Difficulty in making changes — What happens if 
the name of COMM 290 is changed to “Mathe¬ 
matical Optimization”? This would require the 
same change to be made eight times. What if the 
person responsible for making the change for¬ 
gets to change all the sections of COMM 290? 
What then is the “true” name of the course? 

3. Deletion problems — What if there is only one 
section of COMM 290 and it is not offered in a 
particular year? If section 001 is deleted, then the 
system no longer contains any information about 
the course itself, including its name and number 
of credits. 
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FIGURE 3.1: The “monolithic” approach to database design—the catalog view table contains 

information about courses and sections. 


The course “COMM 290” consists 
of many sections. 


Each section has some information 
unique to that section (such as 
Time, Days, Building, 

Room); however, the basic course 
information (e.g., Title, 

Credit s) is the same for all 
sections of a particular course. 
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3. Relationships 

4. Addition problems — If a new section is added to 
any course, all the course information has to be 
typed in again. Not only is this a waste of time, it 
increases the probability of introducing errors into 
the system. 

3.1.1 “Normalized” table design 

The problems identified above can be avoided by 
spitting the Catalog view table into two separate 
tables: 

1 . Courses — information about courses only 

2. Sections — information about sections only. 

The key to making this work is to specify a relation¬ 
ship between courses and Sections so that when 
we look at a section, we know which course it 
belongs to (see Figure 3.2). Since each course can 
have one or more sections, such a relationship is 
called “one-to-many”. 


Introduction: The advantage of using tables and relation- 

FIGURE 3.2: A one-to-many relationship between 

Courses and Sections. 


| Courses 


Sections 

DeptCdde 

1 CO 


CrsNum 

1 CO 

CrsNum 

Title 


Section 

Credits 


Session 

Activity 


CatalogNum 



Term 


Access uses relationships in the following way: 
Assume you are looking at Section 004 of 
COMM 290. Since Dept and CrsNumare included in 
the Sections table, and since a relationship line 
exists between the same two fields in the courses 
table, Access can trace back along this line to the 
Courses table and find all the course-specific infor¬ 
mation. All other sections of COMM 290 point back 
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3. Relationships 

to the same record in the courses table so the 
course information only needs to be stored once. 

3.2 Learning objectives 

□ Why do I want to represent my information in 
multiple tables connected by relationships? 

□ How do I create relationships in Access? 

□ How do I edit or change relationships? 

□ What is referential integrity and why is it 
important? 

3.3 Tutorial exercises 

3.3.1 Creating relationships between 
tables 

• Close the catalog view table and return to 
the database window. 


Learning objectives 

• Select Tools > Relationships from the main 
menu. 

A in version 2.0 the menu structure is slightly 
. different. As such, you select Edit > Relation¬ 
ships instead. 

• To add a table to the relationship window, select 
Relationships > Show Table from menu or press 
the show table icon (%J) on the tool bar. 

• Perform the steps shown in Figure 3.3 to add the 
Courses and Sections tables. 

• Specify the relationship between the primary 
key in courses and the foreign key in Sec¬ 
tions. This is shown in Figure 3.4. 

A Do not check cascading deletions or updates 
. unless you are absolutely sure what they 
mean. See on-line help if you are curious. 
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3. Relationships 


Tutorial exercises 


FIGURE 3.3: Add the Courses and Sections tables to the relationship window. 


© 


© 


The rectangular “field list” represents a 
table. Note that the key (or keys) composing 
the primary key are shown in bold type. 


Relationships 


Select the table you wish to add and either 
double-click or press Add. Repeat as necessary. 


| Courses 


Courses_1 

DeptCode 

s* 

TBSSHSS^B 

CrsNum 


CrsNum 

Title 


Title 

Cret^^ 


Credits 

MiviV 


Activity 


If you accidently add a table more than once, it 
will show up with a <table name>_l label. 
To delete the extra version, click anywhere on 
the unwanted rectangle and press the delete key. 


Show Table 


Tables 


Catalog yiew 
Cours< 

Departments 
Employees 


Sections 




Both 


7 


Add 


Close 
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Tutorial exercises 


FIGURE 3.4: Create a relationship between the two tables. 


© 


Select the primary key 
on the “one” side of the 
relationship. 


-I Relationships 


\ 


d 


To select a concatenated 
key (more than one 
field) hold down the 
Control key while 
selecting. 

Ensure that the correct 
fields are associated 
with each other (this 
must be done manually 
for concatenated keys). 

Check the box to 
enforce referential 
integrity. 


Courses 


DeptCode 

CrsNum 


Tit' 

Cr 

■ A,c Table/Query 


Sections 

DeptCode ^ 

3" 

CrsNum 
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Relationships 


Re I ate d Tab I e / Q u e ry: 
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Cascade Update Rel 
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CrsNum 


Section 

Session 

CatalogNum 

Term 

Days 

Time 


Hi 

■ 
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Relationship Type: 


One-T o-Many 


Drag the selected fields on to the 
foreign key on the “many” side of the 
relationship. 


Create 


© 


Cancel 



Join Type... 


Z Relationships 


If done 
correctly, the 
connectivity (1 
to oo ) shows on 
the relationship 
line(s). 


| Courses 

r 
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1 CSO 
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Section 
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Session 
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CatalogNum 
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Discussion 


3. Relationships 


3.3.2 Editing and deleting relationships 

There are two common reasons for having to edit or 

delete a relationship: 

1. You want to change the data type of one of the 
fields in the relationship — Access will not let you 
do this without first deleting the relationship (after 
you change the data type, you must re-create the 
relationship). 

2. You forget to specify referential integrity — if the 
“1” and symbols do not appear on the rela¬ 
tionship line, then you have not checked the box 
to enforce referential integrity. 

In this section, assume that we have forgotten to 

enforce referential integrity between courses and 

Sections. 

• Perform the steps shown in Figure 3.5 to edit the 
relationship between courses and Sections. 



Note that simply deleting the table in the rela¬ 
tionship window does not delete the relation¬ 
ship, it merely hides it from view. 


3.4 Discussion 


3.4.1 One-to-many relationships 

There are three types of relationships that occur in 

data modeling: 

1 . one-to-one — A one-to-one relationship exists 
between a student and a student number. 

2. one-to-many — A one-to-many relationship 
exists between courses and sections: each 
course may consist of many sections, but each 
section is associated with exactly one course. 

3. many-to-many — A many-to-many relationship 
exists between students and courses: each stu¬ 
dent can take many courses and each course 
can contain many students. 
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Discussion 


FIGURE 3.5: Edit an existing relationship. 


lb 


Select the relationship by clicking on 
the joining line (click on either line if 
the key is concatenated). If you do 
this correctly, the line becomes 
darker. 


With the relationship selected, right- 
click to get the edit/delete pop-up 
menu. If you do not get this menu, 
make sure you have correctly 
selected the relationship. 


Relationships 


® The missing “1” and “°o” symbols 
indicate that referential integrity has 
not been enforced. 



Sections 


DeptCode ^ 


Edit Relationship. 


Delete Relationship 


CatalogNum 
Term *J 
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3. Relationships 


Discussion 


Although the data modeling technique used most 
often in information system development— Entity- 
Relationship diagraming —permits the specifica¬ 
tion of many-to-many relationships, these relation¬ 
ships cannot be implemented in a relational 
database. As a consequence, many-to-many rela¬ 
tionships are usually broken down into a series of 
one-to-many relationships via “composite entities” 
(alternatively, “bridging tables”). Thus to implement 
the student-takes-course relationship, three tables 
are used: Students, Courses, and Student- 
TakesCourse. 

3.4.2 Referential integrity 

One important feature of Access is that it allows you 
to enforce referential integrity at the relationship 
level. What is referential integrity? Essentially, refer¬ 
ential integrity means that every record on the 


“many” side of a relationship has a corresponding 
record on the “one” side. 

Enforcing referential integrity means that you cannot, 
for instance, create a new record in the Sections 
table without having a valid record in the Courses 
table. This is because having a section called 
“BSKW 101 Section 001” is meaningless unless 
there is a course called “BSKW 101”. In addition, ref¬ 
erential integrity prevents you from deleting records 
on the “one” side if related records exist on the 
“many” side. This eliminates the problem of 
“orphaned” records created when parent records are 
deleted. 

Referential integrity is especially important in the 
context of transaction processing systems. Imagine 
that someone comes into your store, makes a large 
purchase, asks you to bill customer number “123”, 
and leaves. What if your order entry system allows 
you to create an order for customer “123” without 
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Application to the assignment 


first checking that such a customer exists? If you 
have no customer 123 record, where do you send 
the bill? 

In systems that do not automatically enforce referen¬ 
tial integrity, these checks have to be written in a pro¬ 
gramming language. This is just one example of how 
table-level features can save you enormous pro¬ 
gramming effort. 

A Enforcing referential integrity has obvious 
. implications for data entry: You cannot popu¬ 
late the “many” side of the table until you pop¬ 
ulate the “one” side. 


A A primary key and a foreign key must be of 
La the same data type before a relationship can 
be created between them. Because of this, it 
is important to remember that the autonumber 
data type (or counter in version 2.0) is really a 
long integer. 

K it never makes sense to have a relationship 
. between two autonumber fields. A foreign key 
cannot be an autonumber since referential 
integrity constraints require it to take on a an 
existing value from a parent table. 


3.5 Application to the assignment 

• Specify all relationships—including referential 
integrity constraints—between tables in your sys¬ 
tem. You are not responsible for cascading 
updates/deletions in this assignment. 
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Access Tutorial 4: Basic Queries Using QBE 


4.1 Introduction: Using queries to 
get the information you need 

At first glance, it appears that splitting information 
into multiple tables and relationships creates more of 
a headache than it is worth. Many people like to 
have all the information they need on one screen 
(like a spreadsheet, for instance); they do not want to 
have to know about foreign keys and relationships 
and so on. 

Queries address this problem. They allow the user to 
join data from one or more tables, order the data in 
different ways, calculate new fields, and specify cri¬ 
teria to filter out certain records. 

The important thing is that the query itself contains 
no data —it merely reorganizes the data from the 
table (or tables) on which it is built without changing 
the “underlying tables” in any way. 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 25-Aug-1997 


Once a query is defined, it can be used in exactly the 
same way as a table. Because of this, it is useful to 
think of queries as “virtual tables”. Similarly, in some 
DBMSes, queries are called “views” because they 
allow different users and different applications to 
have different views of the same data. 

4.2 Learning objectives 

□ Do queries contain any data? 

□ How do I create a query? 

□ What can I do with a query? 

□ How do I create a calculated field? 

□ Why does Access add square brackets 
around field names? 

□ What names should I give the queries I 
create? 

□ What does the ampersand operator (&) do? 
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□ What is a non-updatable recordset? How do I 
tell whether a query results in a non- 
updatable recordset? 

4.3 Tutorial exercises 

4.3.1 Creating a query 

• Use the New button in the Queries pane of the 
database window to create a new query as 
shown in Figure 4.1 . 

• Add the Courses table to the query as shown in 
Figure 4.2. 

• Examine the basic elements of the query design 
screen as shown in Figure 4.3. 

• Save your query (Control-S) using the name 

qryCourses. 


4.3.2 Five basic query operations 


4.3.2.1 Projection 

Projecting a field into a query simply means includ¬ 
ing it in the query definition. The ability to base a 
query on a subset of the fields in an underlying table 
(or tables) is particularly useful when dealing with 
tables that contain some information that is confiden¬ 
tial and some that is not confidential. For instance, 
the Employees table you created in Tutorial 2 con¬ 
tains a field called salary. However, most of the 
queries seen by end-users would not include this 
information, thereby keeping it private. 

• Perform the steps shown in Figure 4.4 to project 
the Deptcode, crsNum, and Title fields into 
the query definition. 

• Select View > Datasheet from the menu to see 
the results of the query. Alternatively, press the 
datasheet icon ( m) on the tool bar. 


\4tH ome I Previous] 


I Next ► 1 


2 of 27 





















4. Basic Queries Using QBE 
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FIGURE 4.1: Create a new query. 


© 


Select the Queries tab in 
the database window. 


ipunivO v7 : Database 


Queries 



F ortn s | © Rep □ rts | S Maero s | Mo dulesj 

jDpen 


Press the New button to 
create a new query. 


New Query 


Avoid the use of the query wizard 
at this point. Queries are very 
important and it is best to learn to 
create them from scratch. 


Create a new query without 
using a wizard. 


n 


New 



Desiqn View 


Simple Query Wizard 
Cro s stab Q u e ry Wi z ard 
Find Duplicates Query Wizard 
Fi n d U n mi atch e d Q u e ry Wi z ard 


OK 


Cancel 
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FIGURE 4.2: Add tables to your query using the “show table” window. 


Show Table 


Tables 


Queries 


Both 


Catalog View 


Courses 


Departments 
Em payees 
Sectinns 



Add 


Close 


Add the Courses table to the query 
by selecting it and pressing Add 
(alternatively, you can simply double¬ 
click on the table you want to add). 







1 

Press Close when done (the “show 
table” window is “modal”—you can 


© 


not do anything else in Access until a 
modal window is closed). 


Microsoft Access 


File Edit View Insert 


a # 



Query 


Tools Window Help 




Select 
Crosstab 
Make Table... 
Update 
Ajjptnd... 
Delete 


The “show table” window is always 
available from the Query > Show Table 
menu. Alternatively, you can press the 
“show table” button on the tool bar. 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.3: The basic elements of the query design screen. 


as 1 Queryl : Select Query 
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which the 
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Title 
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If you “lose” tables in the top 
pane, you have to use the 
horizontal and vertical scroll 
bars to return to the upper-left 
corner of the pane. 



u 




mmmm. 
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The lower 

/ 

pane contains 


the actual 


query 


definition. 



Field 

Table 

Sort 

Show 

Criteria 


DeptCode 


Courses 



Field row— shows the name of the 
fields included in the query. 





v 






hi [_x_ 


T~ZL 


Table row— shows the name of the 
table that the field comes from. To get 
table names in version 2.0, select View 
> Table Names from the menu. 


Criteria row — allows you 
to specify criteria for 
including or excluding 
records from the results set. 


Show boxes— determine 
whether fields included 
in the query are actually 
displayed. 


Sort row— allows you to specify the 
order in which the records are 
displayed 
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FIGURE 4.4: Project a subset of the available fields into the query definition. 


® To project all the fields in the 
Courses table (including 
any that might be added to the 
table after this query is 
created) drag the asterisk (*) 
into the query definition grid. 



To save time when 
projecting fields, select more 
than one field at once (by 
holding down the Control 
key) and dragging all the 
fields as a group. 


Select the field you wish to project and 
drag it into the query definition grid. 
Alternatively, double-click the field. 


ai 0 qryCourses : Select Query 
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4. Basic Queries Using QBE 


Tutorial exercises 


Select View > Query Design to return to design 
mode. Alternatively, press the design icon (M ) 
on the tool bar. 


4.3.2.2 Sorting 

When you use a query to sort, you do not change the 
physical order of the records in the underlying table 
(that is, you do not sort the table). As a result, differ¬ 
ent queries based on the same table can display the 
records in different orders. 

• Perform the steps shown in Figure 4.5 to sort the 
results Of qryCourses by DeptCode and 
CrsNum. 



Since a query is never used to display data to 
a user, you can move the fields around within 
the query definition to get the desired sorting 
precedence. You then reorder the fields in the 
form or report for presentation to the user. 


4.3.2.3 Selection 

You select records by specifying conditions that each 
record must satisfy in order to be included in the 
results set. In “query-by-example” you enter exam¬ 
ples of the results you desire into the criteria row. 

• Perform the steps shown in Figure 4.6 to select 
only those courses with a DeptCode = "comm". 

4.3.2.4 Complex selection criteria 

It is also possible to create complex selection criteria 
using Boolean constructs such as AND, OR, and 
NOT. 

• Project the credits field into the query. 

• Perform the steps shown in Figure 4.7 to create a 
query giving the following result: 

“Show the department, course number, and title 
of all courses in the Commerce department for 
which the number of credits is greater than 
three.” 
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FIGURE 4.5: Sorting the results set on one or more fields. 


bs 1 qryCourses : Select 



Courses 


r" 

Jc 


DeptCode 

CrsNum 

Title 

Credits 

Activity 


JU 


Field 

Table 

Sort 

Show 

Criteria 

or 


Select “ascending” for the DeptCode field 
and “descending” for the CrsNum field. 
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View the results and notice 
the order of the records. 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.6: Select a subset of records from the courses table matching a specific criterion. 


sip qryCourses : Select Query 
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Type the expression "COMM" in the criteria row 
of the Dept Code field. You could also type 
= "COMM" but the equal sign is always implied 
unless another relational operator is used. 


&p qryCourses : Select Query 




View the results. Only records 
matching the criteria are shown. 
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FIGURE 4.7: Select records using an AND condition. 
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&P qryCourses : Select Query 
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d Show the result. 


agp qryCourses : Select Query 


When multiple criteria are placed in the 
same row, they are AND-ed. In other 
words, the records in the results set 
must satisfy DeptCode = "COMM" 
AND Credits > 3. 


Department Course number 


Title 

Applied Statistics in E 



© 


Field 

Table 

Sort 

Show 

Criteria 

or 


Note that the number 
3 is not in quotation 
marks whereas the 
string of characters 
“COMM” is. 


Enter the first criteria: 

"COMM" 



In the same row, enter the second 

> 3 


Uncheck the “show” 
box (Credits is 
used as a criterion but 
it is not displayed in 
the results set) 
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4. Basic Queries Using QBE 


Tutorial exercises 


• Perform the steps shown in Figure 4.8 to create a 
query giving the following result: 

“Show the department, course number, and title 
of all courses from the Commerce department 
and also show those from the Creative Writing 
department for which the number of credits is 
greater than three.” 

4.3.2.5 Joining 

In Tutorial 3, you were advised to break you informa¬ 
tion down into multiple tables with relationships 
between them. In order to put this information back 
together in a usable form, you use a join query. 

• Close qryCourses. 

• Open the relationships window and ensure you 
have a relationship defined between courses 
and Sections. If you do not, create one now (do 
not forget to enforce referential integrity). 

• Create a new query called qryCataiogNum 
based on the Courses and Sections tables. 


• Project Title from the Courses table and 
DeptCode, CrsNum, Section and Catalog- 

Num from the Sections table (see Figure 4.9). 

• Follow the instructions in Figure 4.10 to move 
CataiogNum to the far left of the query definition 
grid. 

Access performs an automatic lookup of information 
from the “one” side of the relationship whenever the 
a valid value is entered into the foreign key of the 
“many” side of the relationship. To see how this 
works, create a new section of “MUSC 105”: 

• Scroll to the bottom of the query in datasheet 
mode and click on the department field. 

• Enter “MUSC”. 

• Enter “105” in the course number field. 

Once Access knows the DeptCode and CrsNum of 
a section, it can uniquely identify the course that the 
section belongs to (which means it also knows the 
values of Title, Credits, Activity, etc.) 
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4. Basic Queries Using QBE Tutorial exercises 

FIGURE 4.8: Select records using an AND and an OR condition. 


© 


When multiple criteria are placed in 
different rows, then they are OR-ed. In 


must satisfy DeptCode = 
OR (DeptCode = "CRWR" 
Credits > 3). 


"COMM" 

AND 



uepiLAJue 


CrsNum 


Title 


Credits 


Activity 


Enter the 
DeptCode 
criteria in 
different rows. 



Department 

Course number 

Title 

► 

MHM1 

290 

Introduction to Guantative Decision |y 


COMM 

291 

Applied Statistics in Business 


COMM 

351 

Financial Accounting 


COMM 

439 

Ad va ncedTopicsin In f o rm ati o n S y st 


CRWR 

202 

Creative Forms 


CRWR 

496 

Poetry T u&- Enter the Credits 

¥ 



E/lllvI lllv v^l vLlllu 


second row. 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.9: Create a query that joins Courses and Sections. 



si 1 c 

iryCatalogNum : Select Query 




Title 

Departmen 

Course nur 

Section 

CatalogNur 

► 

Introduction COMM 

290 

001 

44411 


Introduction 

COMM 

290 

002 

57455 


Introduction 

COMM 

290 

003 

48516 


Introduction 

COMM 

290 

004 

71845 


Introduction 

COMM 

290 

005 

69495 


Introduction 

COMM 

290 

006 

34134 


Introduction 

COMM 

290 

007 

45938 


Introduction 

COMM 

290 

008 

27839 


Applied Stat 

COMM 

291 

001 

84203 


Applied Stat 

COMM 

291 

002 

83920 


Field 

Table 

Sort 

Show 

l' I + . V I ■=. 


Title 


Courses 




DeptCode 


Sections 




Project fields from both tables into 
the query definition. 


CrsNum 


Sections 




Section 


Sections 





CatalogNum 


Sections 





■ 


1 A Home 1 

B 

Previous | 

1 13 Of 27 | 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.10: Move a field within the query definition grid. 


§jp qryCatalogNum : Select Query 


Courses 


DeptCode 

CrsNum 

Title 

Credits 

Activity 


1 DO 


1 DO 


Sections 

* 

DeptCode 

jfc. 

CrsNum 


Section 


Session 


| CatalogNum ▼ 11 


■ : : 


Field 

Table 

Sort 

Show 

Criteria 

or 


hi I 


mm 








Title 

DeptCode 

CrsNum 

Section 

Courses 

Sections 

Sections 

Sections 






v' 

v' 

v' 








\ 







Click once on the grey 
“column selector” 
above the field you 
want to move (if 
properly selected, the 
column turns black). 


To delete a field from 
the query definition, 
select it and press the 
Delete key. 


Drag the selected column to 
its new location. 
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Tutorial exercises 


4. Basic Queries Using QBE 

4.3.3 Creating calculated fields 

A calculated field is a “virtual field” in a query for 
which the value is a function of one or more fields in 
the underlying table. To illustrate this, we will create 
two calculated fields: 

1. one to combine Deptcode and crsNum into one 
field, 

2. one to translate the credits field into a dichoto¬ 
mous string variable (full year or half 
year). 

The syntax of a calculated field is always the same: 

<calc field name>: <definition> 

For example, the syntax for the calculated field 
called Course is: 

Course: DeptCode & CrsNum 

The calculated field name can be just about any¬ 
thing, as long as it is unique. The definition is any 
expression that Access can evaluate. In this case, 


the expression involves two fields from the Courses 
table (Deptcode and crsNum) and the ampersand 
operator (see Section 4.4.2 for more information on 
using the ampersand operator). 

• Create a new query called qryCourseLengths 
based on the courses table. 

• Follow the instructions in Figure 4.11 to create 
the calculated field course 

• Run the query to verify the results, as shown in 
Figure 4.12. 

A When you use field names in expressions, 

. Access normally adds square brackets. This 
is not cause for concern because in Access, 
square brackets simply indicate the name of a 
field (or some other object in the Access envi¬ 
ronment). However, if your field name con¬ 
tains blank spaces (e.g., Dept code), the 
square brackets are NOT optional—you must 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.11: Create a calculated field based on two other fields. 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.12: The resulting calculated field. 


© 


When the zoom window is 
closed, Access adds square 
brackets to the field names. 
Since the field names in this 
example do not contain 
spaces, the brackets are 
optional. 


ai 1 qryCourseLengths : Select Query 


© 


Courses 


DeptCode 

CrsNum 

Title 

Credits 

Activity 


The name of the 
calculated field shows in 
the field selector. 


is 1 qryCourseLenc /is : Select Query 


M 


Fidcf 

Table 

Sort 

Show 

Criteria 

or 


-1— 

'Course: [DeptCode' 

T 







)/ 




Course 


COMM291 

COMM351 

COMM439 

CRWR202 

CRWR496 

EDUC306 

A 


u 




The ampersand operator (&) simply tacks 
CrsNum onto the end of DeptCode. 
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4. Basic Queries Using QBE 

type them every time you use the field name 
in an expression. 

4.3.3.1 Refining the calculated field 

Instead of having DeptCode and CrsNum run 
together in the new course field, you may prefer to 
have a space separating the two parts. 

• Edit the Courses field by clicking on the field row 
and invoking the zoom box. 

• Add a space (in quotation marks) between the 
two constituent fields: 

Course: DeptCode & " " & CrsNum 

• Switch to datasheet mode to see the result. 

4.3.3.2 A more complex calculated field 

To create a calculated field that maps credits to a 
dichotomous string variable, we need a means of 
testing whether the value of credits exceeds a 
certain threshold (e.g., any course with more than 


Tutorial exercises 

three credits is a full-year course). To do this, we will 
use the “immediate if” (iif) function. 

• Search on-line help for information about the 
iif () function. 

Basically, the function uses the following syntax: 

iif (<expression>, <true part>, 

<false part>) 

to implement the following logic: 

IF <expression> = TRUE THEN 
RETURN <true part> 

ELSE 

RETURN <false part> 

END IF 

• Create a new calculated field called Length: 

Length: iif(Credits > 3, "full 
year", "half year") 

• Verify the results, as shown in Figure 4.13. 
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4. Basic Queries Using QBE 


Tutorial exercises 


FIGURE 4.13: Create a calculated field using the “immediate if” function 


Create a calculated field called Length with the following expression: 

Length: iif(Credits>3, "full year", "half year") 


Courses 


DeptCdde 

CrsNum 

Title 

Credits 

Activity 


SI Zoom 


Length: llf([Credits]>3 l M full year'V'half year 11 ) 


JJ 


Field: 

Table: 

Sort: 

Show: 

Criteria: 

or: 


Course: [PeptCodE 


v' 


as 1 c 

jryCourse Lengths 

: Select Query 


Course 

Length 

► 

COMM 290 

half year 


COMM 291 

full year 


COMM 351 

half year 


COMM 439 

half year 


CRVVR 202 

full year 


CRVVR 496 

full year 


EDUC 306 

half year 


ENGL 301 

half year 


MATH 303 

half year 


MATH 407 

half year 


MUSC 105 

half year 

* 




I '^Homel N Previous | 19 of 27 | Next ► 1 


4. Basic Queries Using QBE Discussion 


4.3.4 Errors in queries 

It may be that after defining a calculated field, you 
get the “enter parameter” dialog box shown in 
Figure 4.14 when you run the query. This occurs 
when you spell a field name incorrectly. Access can¬ 
not resolve the name of the misspelled field and thus 
asks the user for the value. To eliminate the problem, 
simply correct the spelling mistake. 

FIGURE 4.14: A spelling error in a calculated 

field. 


Enter Parameter Value 


Creditz Access cannot find the 

I ^- field named Creditz 


Cancel 


4.4 Discussion 

4.4.1 Naming conventions for database 
objects 

There are relatively few naming restrictions for data¬ 
base objects in Access. However, a clear, consistent 
method for choosing names can save time and avoid 
confusion later on. Although there is no hard and fast 
naming convention required for the assignment, the 
following points should be kept in mind: 

• Use meaningful names — An object named 
Tabiei does not tell you much about the con¬ 
tents of the table. Furthermore, since there is no 
practical limit to the length of the names, you 
should not use short, cryptic names such as 
s9 6w_b. As the number of objects in your data¬ 
base grows, the time spent carefully naming your 
objects will pay itself back many times. 


OK 
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4. Basic Queries Using QBE 


Discussion 


• Use capitalization rather than spaces to separate 
words — Unlike many database systems, Access 
allows spaces in object names. However, if you 
choose to use spaces, you will have to enclose 
your field names in square brackets whenever 
you use them in expressions (e.g., [Back 
Orders] ). As such, it is slightly more efficient to 
use a name SUCh as Backorders than Back 
Orders. 

• Give each type of object a distinctive prefix (or 
suffix) — This is especially important in the con¬ 
text of queries since tables and queries cannot 
have the same name. For example, you cannot 
have a table named Backorders and a query 
named Backorders. However, if all your query 
names are of the form qryBackOrders, then 
distinguishing between tables and queries is 
straightforward. 


4. Basic Queries Using QBE 

Table 4.1: A suggested naming convention for 
Access database objects. 


Object type 

Prefix 

Example 

table 

(none) 

OrderDetails 

query 

qry 

qryNonZeroBackOrders 

parameter 

query 

pqry 

pqryItemsInOrder 

form 

frm 

frmOrders 

sub form 

sfrm 

sfrmOrderDetails 

switchboard 

form 

swb 

swbMainSwitchboard 

report 

rpt 

rptInvoice 

sub report 

srpt 

srptInvoiceDetails 

macro 

mcr 

mcrOrders 

Visual Basic 
module 

bas 

basUtilities 


• Stick to standard alphanumeric characters — You 
should limit yourself to the characters [A...Z], 
[a...z], [0...9], and perhaps underscore (_) and 
dash (-). Although Access allows you to use virtu¬ 
ally any character, undocumented problems have 
been encountered in the past with non-alphanu- 
meric characters such as the pound sign (#). 

Table 4.1 shows a suggested naming convention for 
Access database objects (you will discover what 
these objects are in the course of doing the tutorials). 

4.4.2 The ampersand (&) operator 

The ampersand operator is like any other operator 
(e.g., +, -, X, -r) except that it is intended for use on 
strings of characters. What the ampersand does is 
simply add one string on to the end of another string 
(hence its other name: the “concatenation” operator). 
For example, the expression 

"First string" & "Second string" 


I ^HHomel H Previous! 21 of 27 | Next^ | 


Discussion 


yields the result 

First stringSecond string 

However, if a space is include within the quotation 
marks of the second string (" second string"), 
the result is: 

First string Second string 

4.4.3 Using queries to populate tables 
on the “many” side of a 
relationship 

In Section 4.3.2.5, you added a record to the Sec¬ 
tions table to demonstrate the automatic lookup 
feature of Access. However, a common mistake 
when creating queries for entering data into tables 
on the “many” side of a relationship is to forget to 
project the table’s foreign key. That is, faced with two 
tables containing the fields Deptcode and crsNum, 
you project the fields from the wrong table (the “one” 
side) into your query definition. 
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4. Basic Queries Using QBE 


Discussion 


To illustrate the problem, do the following: 

• Open the qryCataiogNum query and make the 
changes shown in Figure 4.15. 

• Attempt to save the new section of “MUSC 105” 
as shown in Figure 4.16. 

There are two ways to avoid this error when deciding 

which fields to project into your join queries: 

1. Always show the table names when creating a 
query based on more than one table. That way, 
you can quickly determine whether the query 
makes sense. 

2. Always ask yourself: “What is the purpose of this 
query?” If the answer is: “To add new records to 
the Sections table,” you automatically have to 
include a//the fields from the Sections table. 
Fields from the Courses table are only shown 
for validation purposes. 


4.4.4 Non-updatable recordsets 

Another problem that sometimes occurs when creat¬ 
ing join queries is that the query is not quite right in 
some way. In such cases, Access will allow you to 
view the results of the query, but it will not allow you 
to edit the data. 

In this section, will look at a nonsensical query that 
results from an incompletely specified relationship. 
As you will probably discover, however, there are 
many different way to generate nonsensical queries. 

• Create a new query called qryNonUpdate 
based on the Courses and Sections tables. 

• Delete the crsNum relationship but leave the 
Deptcode relationship intact, as shown in 
Figure 4.17. 

The result of this query is that every section in a 
Commerce course will be associated with every 
Commerce course. Since allowing the user to update 
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4. Basic Queries Using QBE Discussion 

FIGURE 4.15: Create a data-entry query without a foreign key. 


si 1 qryCataiogNum : Select Query 




Reorder the fields (by 
dragging and dropping) so 
that DeptCode and 
CrsNum are on the far left. 


Change the source table for 

DeptCode and CrsNum 
from Sections to 
Courses. 


Switch to datasheet mode 
and attempt to add a new 
section of “MUSC 105”. 


Courses 



1 

Sections 

, 

1 CM3 

DeptCode 

CrsNum 

A 

1 DO 



— 

Section 

Session 


CatalogNum 

d 


© 


In version 2.0 you have to 
select View > Table 
Names to display the 
table row. 




—^- 




A 1 

DeptCode 

-CrsNum 

CataloqNum 

Title 

Section 

Courses 

Courses 

Sections 

Courses 

Sections 






v' 

v' 

v/ 

v' 

v' 











▼ 1 
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4. Basic Queries Using QBE 


Discussion 


FIGURE 4.16: The result of attempting to save a record in which the foreign key is missing 


as 0 qryCatalogNum : Select Query 



Departmen 

Course nur 

CatalogNur 

Title 

Section 


MUSC 

105 n 

84545 1 

Aural Skills 

003 


COMM 

439 

57167 

Advanced T 

001 


CRVVR 

202 

28456 

Creative For 

001 


CRVVR 

202 

38804 

Creative For 

901 



CRVVR 

202 

00834 

Creative For 

902 

4 

8 

L 

MUSC 

105 

1 Microsoft Access 

1 

r 



/ f'\ Duplicate value in ind ex primary ke 



Attempt to save the 
new section by 
clicking its record 
selector. 


Since the fields are bound to the 
Courses table, you are 
attempting to replace the 
current record in the Courses 
table with “MUSC 105”. But 
since a “MUSC 105” already 
exists, you get an error. 


OK 


Help 
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4. Basic Queries Using QBE 


Discussion 


FIGURE 4.17: Create a non-updatable recordset. 


si 1 qryNonUpdate : Select Query 


| Courses 


Sections 

■Ji 

DeptCode 

1 DO 

* 

DeptCode 


CrsNum 

J 

CrsNum 


Title 

4 

Section 


Credits 

/ 

Session 


Activity A 


|CatalogNum ▼ 11 


Project fields from both tables and 
view the query in datasheet mode 
(i.e., view the “recordset”). 


jj 



LJ 


To create a nonsensical query, delete the 
Cr sNum relationship by clicking on it 
and pressing the Delete key. Leave the 
Dept Code relationship intact. 


© 





0eptCode 

CrsNum 

Section 

'Courses 

Courses 

Sections 














c 

fiyNonUpdate : Select Query 




Department cc 

Course numbe 

Section 


COMM [437 

001 ^ 


COMM 

437 

002 


COMM 

437 

_ A tfpmn 

it tr* 


COMM 

437 ^ 1 

change a value in 
the recordset. 


COMM 

437 


COMM 

437 

'W r 



COMM 

437 

003 



COMM 

437 

001 


He cord: 

±Lia 

1 1 



1 ► | H | | of 108 


Note the absence of the asterisk and the “new record” 
row. This is a sure sign that the recordset is non-updatable. 
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4. Basic Queries Using QBE 


Application to the assignment 


the values in this recordset would create anomalies, 
Access designates the recordset as non-updatable. 

A A common mistake is to build data entry 
. forms on nonsensical queries and to assume 
that there is a mistake in the form when the 
forms do not work. Clearly, if a query is non- 
updatable, a form based on the query is also 
going to be non-updatable. A quick check for 
a “new record” row in the query can save time 
and frustration. 

4.5 Application to the assignment 

• Create a query to sort the Products table by 

Product ID. 

• Create a query that joins the orderDetaiis 
and Products tables. When you enter a valid 
Product id, the information about the product 
(such as name, quantity on hand, and so on) 


should appear automatically. If they do not, see 
Section 4.4.3. 

• Create a calculated field in your qryOrderDe- 
taiis query that calculates the extended price 
(quantity shipped x price) of each order detail. 

• Enter the first order into your system by entering 
the information directly into tables or queries. 
This involves creating a single Orders record 
and several orderDetaiis records. You must 
also consult the Products and Backorders 
tables to determine the quantity of each item to 
ship. 



Entering orders into your system will be much 
less work once the input forms and triggers 
are in place. The goal at this point is to get 
you thinking about the order entry process 
and ways in which it can be automated. 
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Access Tutorial 5: Basic Queries using SQL 


5.1 Introduction: The difference 
between QBE and SQL 

Query-By-Example (QBE) and Structured Query 
Language (SQL) are both well-known, industry-stan¬ 
dard languages for extracting information from rela¬ 
tional database systems. The advantage of QBE (as 
you saw in Tutorial 4) that it is graphical and rela¬ 
tively easy to use. The advantage of SQL is that it 
has achieved nearly universal adoption within the 
relational database world. 

With only a few exceptions (which you probably will 
not encounter in this assignment) QBE and SQL are 
completely interchangeable. If you understand the 
underlying concepts (projection, selection, sorting, 
joining, and calculated fields) of one, you understand 
the underlying concepts of both. In fact, in Access 
you can switch between QBE and SQL versions of 
your queries with the click of a mouse. 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 22-Aug-1997 


Although you normally use QBE in Access, the ubiq¬ 
uity of SQL in organizations necessitates a brief 
overview. 

5.2 Learning objectives 

□ What is the difference between QBE and 
SQL? 

□ How do I create an SQL query? 

5.3 Tutorial exercises 

In this section, you will create a few simple queries in 
SQL. 

• Create a new query but close the “show table” 
dialog box with out adding tables. 

• Select View> SQL to switch to the SQL editor as 
shown in Figure 5.1 . 
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5. Basic Queries using SQL 


Tutorial exercises 


FIGURE 5.1: Open a query in SQL mode 


Microsoft Access 


View 


File Edit 


Insert Query Tools Window Help 
Query Design 


SQL 


L 

|ijit§) 

ip 

A 

2J 


m 


Datasheet 


Totals 

* Table Names 


21 


as 8 Queryl : Select Query 


SELECT DISTINCTROW; 


5.3.1 Basic SQL queries 

A typical SQL statement resembles the following: 

SELECT DeptCode, CrsNum, Title FROM 
Courses WHERE DeptCode = "COMM"; 

There are four parts to this statement: 

1. SELECT <field 1 , field 2 , ..., field n > ... 

— specifies which fields to project (the dis- 
tinctrow predicate shown in Figure 5.1 is 
optional and will not be discussed in this tutorial); 


2. ... from <tabie> ... — specifies the underlying 
table (or tables) for the query; 

3. ... WHERE tconditiorq AND/OR 
condition 2 , ..., AND/OR condition n > — 

specifies one or more conditions that each record 
must satisfy in order to be included in the results 
set; 

4. ; (semicolon) — all SQL statements must end 
with a semicolon (but if you forget it, Access will 
add it for you). 

These can now be put together to build an SQL 
query: 

• Type the following into the SQL window: 

SELECT DeptCode, CrsNum, Title FROM 
Courses WHERE DeptCode = "COMM"; 

• Select View > Datasheet to view the results. 

• Select View > Query Design to view the query in 
QBE mode, as shown in Figure 5.2. 

• Save your query as qryCoursesSQL. 
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5. Basic Queries using SQL 


Tutorial exercises 


FIGURE 5.2: The SQL and QBE views are interchangeable. 


ai 1 Queryl : Select Query 




SELECT DeptCode, CrsNum, Title FROM Courses 
WHERE DeptCode = "COMM"; 



When you return to SQL mode 
after viewing your query in QBE 
mode, you will notice that Access 
has added some additional text. 
This optional text does not 
change the query in any way 



l^+lomel M Previous | 3 of 5 | Next^ | 


5. Basic Queries using SQL 

5.3.2 Complex WHERE clauses 

You can use AND, OR, and NOT conditions in your 
WHERE clauses in a straightforward manner. 

• Change your query to the following to get all 
Commerce courses with more than three credits: 

SELECT DeptCode, CrsNum, Title 
FROM Courses 

WHERE DeptCode = "COMM" AND Credits 
> 3 

A Note that since DeptCode is a text field, its 
. criterion must be a string (in this case, the lit¬ 
eral string “COMM”). However, Credits is a 
numeric field and its criterion must be a num¬ 
ber (thus, there cannot be quotation marks 
around the 3). 


Tutorial exercises 

5.3.3 Join queries 

Join queries use the same elements as a basic 
select query. The only difference is that the FROM 
statement is replaced with a statement that 
describes the tables to be joined and the relationship 
(i.e., foreign key) between them: 

... FROM table! INNER JOIN table 2 ON 
table 1 .field 1 = table 2 .field 2 ... 

Note that since both tables contain the fields Dept¬ 
Code and CrsNum, the <table name>.<field 
name> notation must be used to remove any ambi¬ 
guity. 

• Create a new SQL query containing the text: 

SELECT Courses.DeptCode, 

Courses.CrsNum, Courses.Title. 
Sections.CatalogNum 

FROM Courses INNER JOIN Sections ON 
Courses.CrsNum = Sections.CrsNum 
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5. Basic Queries using SQL 


Discussion 


AND Courses.DeptCode = 

Sections.DeptCode 

WHERE Courses.DeptCode="COMM"; 

5.4 Discussion 

Although the syntax of SQL is not particularly diffi¬ 
cult, writing long SQL queries is tedious and error- 
prone. For this reason, you are advised to use QBE 
for the assignment. 

In the real world, however, when you say you know 
something about databases, it usually implies you 
know the “data definition” and “data manipulation” 
aspects of SQL in your sleep. If you plan to pursue a 
career in information systems, a comprehensive 
SQL reference book can be a worthwhile investment. 
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Access Tutorial 6: Form Fundamentals 


6.1 Introduction: Using forms as 
the core of an application 

Forms provide a user-oriented interface to the data 
in a database application. They allow you, as a 
developer, to specify in detail the appearance and 
behavior of the data on screen and to exert a certain 
amount of control over the user’s additions and mod¬ 
ifications to the data. 

Like queries, forms do not contain any data. Instead, 
they provide a “window” through which tables and 
queries can be viewed. The relationship between 
tables, queries, and forms is shown in Figure 6.1 . 

In this tutorial, we are going to explore the basic ele¬ 
ments of form creation using Access’ form design 
tools. In subsequent tutorials, we will extend the 
functionality and ease-of-use of our basic forms with 
subforms (Tutorial 7), “combo box” controls 
(Tutorial 8), and triggers (Tutorial 13). 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 24-Aug-1997 


FIGURE 6.1: The relationship between forms, 

queries, and tables. 



6.2 Learning objectives 

□ Do forms contain data? 

□ How do I create a form? 
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6. Form Fundamentals 


Tutorial exercises 


□ How do I make the contents of a field on a 
form read-only? 

□ What is an unbound text box? How do I create 
one? 

□ How do I create a form using the form wizard? 

□ What is the difference between a columnar 
(single-column) and tabular form? 

6.3 Tutorial exercises 

6.3.1 Creating a form from scratch 

Although Access provides an excellent wizard for 
creating simple forms, you will start by building a 
form from scratch. This will give you a better appreci¬ 
ation of what it is the wizard does and provide you 
with the basic knowledge needed to customize and 
refine the wizard’s output. 


• Create a new blank form based on the courses 
table, as shown in Figure 6.2. 

• The basic elements of the design screen are 
shown in Figure 6.3. Use the View mem to dis¬ 
play the toolbox and field list if they are not 
already visible. 

6.3.1.1 Adding bound text boxes 

• Add a “bound” text box for the Deptcode field by 
dragging Deptcode from the field list to the form 
background, as shown in Figure 6.4. 

• Reposition the Deptcode text box in the upper 
left of the form. 



Remember that you can always use the 
“undo” feature to reverse mistakes. Select 
Edit > Undo from the menu or simply press 
ControTZ (this works the same in virtually all 
Windows applications). 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.2: Create a new form to display data from the courses table. 
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New Form 


Select the Forms tab from 
the database window. 



Create a ne\ 
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irrn without using 


lb 


Select Design View (do not 
use the wizard at this point) 


Ch o o s e th e tab I e or q u e ry wh e re 
the object's data comes from: 


OK 


© 


Since you can build a form on top of a table or a 
query, both are shown in this list (here is where a 
meaningful naming convention starts to pay off) 



Design View 


Form Wizard 


AutoForm 

AutoForm 

AutoForm 


Columnar 

Tabular 

Datasheet 


Chart Wizard 
PivotTable Wizard 


Bind the form to the 
Courses table. 



T 

Catalog View 


Courses 



Departments 

Employees 

qryCatalogNum 

qryCourse Lengths 

qryCourses 

Sections 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.3: The basic elements of the form design screen. 
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To change the size of 
the form, drag the edges 
of the detail section. 



DeptCode 
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Title 

Credits 

Activity 


Courses 


The field list — shows the fields 
in the table or query to which the 
form is bound. 
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Microsoft Access 

File Edit 

View 

Insert Format Tools Window 
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Form 


The toolbox — the icons in the 
toolbox are used to create graphical 
items and controls on the form. 


Form Design 
Form 

Datasheet 

Properties 


Field List 


Tab Order 
Code 

^ Euler 
^ Grid 
^ Toolbox 


© 



Page HeadetyFooter 
Fo rm H e ad e r/ Fas^e r 

Toolbars... 


If the field list and toolbox 
are not displayed, use the 
View menu or toolbar icons. 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.4: Create a bound text box for the Deptcode field. 


© 


Access uses the field’s caption property as the default label for the text box. 
If no caption is specified, the field name (e.g., DeptCode) is used. To save 
time editing labels, choose your captions with this feature in mind. 


B1 Forml : Form 


BIB 


1 

© To move an object and its 
label, drag the center of the 
object (the cursor becomes 
a white arrow). To move 
just the object or just the 
label, drag the upper left 
handle (the cursor becomes 
a pointing finger). 
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Drag the highlighted field on 
to the form’s detail section. 
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6. Form Fundamentals 

• Drag the remaining fields on to the form, as 
shown in Figure 6.5 (do not worry about whether 
the fields are lined up perfectly). 

• Select View > Form to see the resulting form. 
Alternatively, press the form view icon (HI). 

• Select View > Form Design or press the design 
view icon (M_) to return to design mode. 

6.3.1.2 Using a field’s properties to protect its 

contents 

Every object on an Access form (e.g., text box, label, 
detail section, etc.) has a set of properties that can 
be modified. In this section, you are going to use the 
Locked and Enabled properties to control the user’s 
ability to change the information in a field. 

• Select the Deptcode text box and right-click to 
bring up its property sheet, as shown in 
Figure 6.6. 


Tutorial exercises 

• Scroll down the property sheet to the Locked 
property and set it to Yes, as shown in 
Figure 6.7. 

• Switch to the form view and attempt to change 
the contents of the Deptcode field. 

A stronger form of protection than locking a field is 
“disabling” it. 

• Return to design mode and make the following 
changes: reset the Locked property to No; set the 
Enabled property to No. 

• Attempt to change the contents of the Deptcode 
field in form view, as shown in Figure 6.8. 

• Save the form as frmCourses. 

6.3.1.3 Adding an unbound text box 

All the text boxes created in the previous section 
were “bound” text boxes—that is, they were bound to 
a field in the underlying table or query. When you 
change the value in a bound text box, you are mak- 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.5: Add the text boxes and switch to form view to see the resulting form. 
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Title 
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Add the remaining 
fields to the form. 
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j 


® Text boxes are simply 

“windows” on to the fields 
in the underlying table. 


II Forml : Form 



Department code: 
Course number: 
Title: 

Credits: 

Activity: 


COMM 
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Introduction to Guantatk 


LEC 


lb 


Select View > Form from the 
main menu to view the form. 


You can add more than one field to the form with one_I \ 

drag-and-drop operation by holding down the Control 
button when selecting the fields from the field list. 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.6: Bring up the property sheet for the Deptcode text box. 


II Forml : Form 





Select the object (e.g., the 
DeptCode text box) for 
which you wish to see the 
properties. When an object 
has been selected, it is 
bordered by six dark 
“handles”. 



Properties 


Build Event... 

Control Wizard. 
Object 

Change T o 

Align 



Right-click once on the selected 

object to get the pop-up menu. 

10 ■ 1 ■ 1 

Select Properties to get the 
property sheet. 


if Text Box: DeptCode 


Format | Data | Event | Other 

Name. DeptCode 

Control Source 
Format 

Decimal Places. 

Input Mask. 

Default Value .. 

Validation Rule 
Validation T ext 
Status BarTe 
Enter Key B©navior 
Allow Auto Correct 




The properties are broken down 
into four groups. To see all the 
properties, select the All tab. 
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Some properties of the text box (such as 
input mask) are inherited from the field 
to which the text box is bound. 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.7: Change the Locked property of 

DeptCode tO Yes. 
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Text Box: DeptCode 


Format I Data I Event I Other 

||_r— 1 1 1 

| 0 Ente r Key 6 e h avi o r 
^ Allow Auto Correct.. 


Default 


Yes 


Always 


Visible. Yes 

Display When 

Enabled. Yes 

Locked. No 

Filtei%bokup 
Auto T a 
Tab Stop . 

Tab Index. 

Snrnll Ram 


Yes 


Yes 
0 

Nnnp 



Use the scroll bar to find 
the Locked property. 


ing the change directly to the data in the underlying 
table. 

It is possible, however, to create objects on forms 
that are not bound to anything. Although you will not 
use many “unbound” text boxes in the assignment, it 
is instructive to see how they work. 

• Create a new empty form bound to the Courses 
table and save it using the name 

frmCoursesUB. 

• Select the text box tool (aw) from the toolbox and 
create and unbound text box, as shown in 
Figure 6.9. 

6.3.1.4 Binding an unbound text box to a field 

The only difference between a bound and an 
unbound text box is that the Control Source property 
of a bound text box is set to the name of a field. In 
this section, you are going to change the unbound 
text box shown in Figure 6.9 to a bound text box. 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.8: Set the Enabled property of DeptCode to No and attempt to change the value in the 

field. 
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Introduction to Ouantativ 
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LEG 


When a form object is disabled, it 
cannot receive the “focus” (that is, 
you cannot put the cursor on it). 


By default, disabled form objects are 
greyed out. To override this feature, 
set the Locked property to Yes and 
the Enabled property to No. 


I4I-h ome I Previous] 10 of 15 | Next ► ] 









































































































































6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.9: Create an unbound text box. 


Select the text box tool from the toolbox. 
The cursor becomes a small text box. 



• Bring up the property sheet for the unbound text 
box. Change its Control Source property from null 
to Deptcode, as shown in Figure 6.10. 

6.3.2 Creating a single-column form 
using the wizard 

Now that you understand the basics of creating and 
modifying bound text boxes, you can rely on the form 
wizard to create the basic layout of all your forms. 

• Create a new form bound to the courses table 
using the form wizard, as shown in Figure 6.11. 

• Use the form wizard to specify the fields you want 
on your form and the order in which they appear, 
as shown in Figure 6.12. Select “columnar” when 
prompted for the form type. 

A “Columnar” forms are called “single column” 

. forms in version 2.0. 
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6. Form Fundamentals Tutorial exercises 

FIGURE 6.10: Set the Control Source property FIGURE 6.11: Create a new form using the form 

of an unbound text box. wizard. 
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to DeptCode. 
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6. Form Fundamentals 


Tutorial exercises 


FIGURE 6.12: Use the form wizard to determine the order of fields on your form. 


Form Wizard 



Which fields do you want on your form? 

You can choose from more than one table or query. 


Tables/Queries: 


Table: Courses 


Available Fields: 


© 


Title 

Credits 



Selected Fields: 
[beptCode 


CrsNum 



© 

© 


to show a field, either double¬ 
click it or press the > button. 


To show all the fields, press the 
» button. 


< 



<< 




Cancel 


Next > 


Finish 


The order in which 
the fields appear in 
this pane is the order 
in which they will 
appear on the form. 
Use the < and « 
buttons to move 
fields back to the 
pane on the left. 
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6. Form Fundamentals 


Discussion 


The primary advantage of the wizard is that it auto¬ 
matically creates, formats, and aligns the bound text 
boxes. Of course, once the wizard has created a 
form, you are free to modify it in any way. 



If you make a mistake when creating a form 
(e.g., you put the fields in the wrong order) it 
is often easier to use the wizard and start over 
than to fix the problem manually. 


6.4 Discussion 


look and behavior of the data. The three different 
types of forms are shown in Figure 6.13. 

6.5 Application to the assignment 

• Use the wizard to create columnar forms for all 
your master tables. Note that in some cases 
(e.g., Backorders) you will want to base the 
form on a join query rather than table in order to 
show important information such as CustName 
and ProductName. 


6.4.1 Columnar versus tabular versus 
datasheet forms 

Columnar forms show one record per page. Tabular 
forms, in contrast, show many records per page and 
are used primarily as subforms. There is also a a 
datasheet form type, but it is seldom used since it 
gives the developer relatively little control over the 
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6. Form Fundamentals 


Application to the assignment 


FIGURE 6.13: The same information displayed as a columnar, tabular, and datasheet form. 
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A datasheet form is identical to the datasheet 
view of a table or query. Since it gives the 
designer very little control over the format of the 
data, it is generally inappropriate for use in an 
end-user application. 
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Access Tutorial 7: Subforms 


7.1 Introduction: The advantages of 

forms within forms 

A columnar/single-column main form with a tabular 
subform is a natural way of representing information 
from tables with a one-to-many relationship. For 
example, the form shown in Figure 7.1 is really two 
forms: the main form contains information about a 
specific course; the subform shows all the sections 
associated with the course. 

In the Courses and Sections example, the foreign 
key (Deptcode and crsNum) provides a link 
between the two forms. This connection allows 
Access to synchronize the forms, meaning: 

• when you move to another course record, only 
the relevant sections are shown in the subform; 

• when you add a new section, the foreign key in 
the Sections table is automatically filled in (in 


fact, there is no need to show Deptcode and 
CrsNum in the subform). 

Although you will quickly learn to take a feature such 
as form/subform synchronization for granted, it is 
worthwhile to consider what this feature does and 
what it would take if you had to implement the same 
feature using a programming language. 

7.2 Learning objectives 

□ What is form/subform synchronization? 

□ How do I create a form/subform combination? 

□ How do I link a form with a subform? 

7.3 Tutorial exercises 

Although there are a number of different ways to cre¬ 
ate a subform within a main form, the recommended 
procedure is the following: 
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7. Subforms 


Tutorial exercises 


FIGURE 7.1: A typical form/subform combination. 


Because a link is established between the main form 
and the subform, only the sections that belong with 
“COMM 351” are displayed in the subform. 


The main part of the form is 
columnar (one record per page) 
and displays information from ^ 
the Courses table. 



IS Courses 


The subform is a separate 
tabular form that displays 
information from the 
Sections table. 
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7. Subforms 


Tutorial exercises 


1. create and save both forms (one columnar, one 
tabular) separately; 

2. drag the subform on to the main form; and, 

3. verify the linkage between the two forms. 

7.3.1 Creating the main form 

• Use the wizard to create a columnar form based 
on the courses table. 

• Rearrange the fields so that they make efficient 
use of the top part of the form, as shown in 
Figure 7.2. 

• Save the form as f rmCoursesMain. 

7.3.2 Creating the subform 

• Use the wizard to create the subform, as shown 
in Figure 7.3 and Figure 7.4. 

• Subforms created by the wizard typically require 
some fine tuning in order to reduce the amount of 


space they occupy. A number of editing issues 
are highlighted in Figure 7.5. 

• Save the form as sfrmSections and close it. 

7.3.3 Linking the main form and subform 

In this section, you are going to return to the main 
form and drag the saved subform from the database 
window to an appropriate position on the main form. 

• Open the main form (f rmCoursesMain) in 
design mode. 

• Select Window > univO_vx: Database to open the 
database window in the foreground. Alternatively, 
you can press the database window icon ( jj ) on 
the tool bar. 

• Perform the steps shown in Figure 7.6 to drag the 
subform on to the main form. 

• The result of the drag-and-drop operation are 
shown in Figure 7.7. The advantage of the drag- 
and-drop method of creating a sub form is that 
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7. Subforms Tutorial exercises 

FIGURE 7.2: Rearrange the text boxes on the main form to make room for the subform. 


Use the wizard to create a 
columnar form based on 

Courses. 


Enter form design mode and 
rearrange the text boxes to 
make room for the subform. 
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Save the form under the name 

frmCoursesMain. 



+ Form Footer 



To move more than one form object at a time, either 
hold down the Shift key when selecting or drag a box 
through the objects (click and drag to create a box). 
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7. Subforms 


Tutorial exercises 


FIGURE 7.3: Use the wizard to create the Sections subform (part 1). 
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creates your form, based on, 
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7. Subforms 


Tutorial exercises 


FIGURE 7.4: Use the wizard to create the Sections subform (continued) 


Form Wizard 


Wh at I ay o ut wo u I d y o u I i ke f o r y o u r f o rrn ? 


Select Tabular 
layout. 


Cancel 


r Columnar 
** jTabuiar; 
r Datasheet 


Form Wizard 



© 


Select Modify the form’s design to 
enter form design mode directly. 


© 


In version 7.0, the title appears in the bar 
across the top of the form’s window. In 
version 2.0, however, the wizard creates a 
title in a form header. As such, you 
should ensure this is blank if you are 
using version 2.0. 


Wh at titl e d o yo u want f o r yo u r f o rm ? 


\* 

d 


Since a subform is embedded in a main 
form, you do not have to provide a title. 


That's all the information the wizard needs to create your form. 

Do you want to open the form or modify the form's design? 

( * Open the form to view or enter information. 
f* jModify the form's design. 


ome ]E Previous] 6 of 19 | Next ► ] 







































































































7. Subforms 


Tutorial exercises 


FIGURE 7.5: Edit the subform to reduce the amount of space it uses. 


Reduce the horizontal space used 
by the headings and fields. 


Reduce the vertical space by moving the fields up to the 
“detail band” and bringing the “form footer” band up 
against the fields (to move a band, drag it using the mouse). 


il Forml : Form 
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press Shift-Enter. 
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To move all the fields at once, 
drag a “selection box” so that it 
touches each field. Note that the 
box does not have to enclose 
objects for them to be selected. 
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Tutorial exercises 


FIGURE 7.6: Drag the subform on to the main form. 


B1 frmCoursesMain : Form 



Open the main form 
in design mode. 


I 10 1 l ' 1 1 l 12 1 1 '13' 1 14' 1 


Credits 
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Position the database 
window so that the 
subform’s target 
destination is visible. 


univO v7 : Database 


HI Tables | Up Queries 


8j Forms 1 
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frmCourses 

frmCoursesMain 

frmCoursesUB 


unm 

Modules 
Open 


Design 


sfrmSections 


Drag the subform on 
to the main form. 


New 
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7. Subforms 


Tutorial exercises 


the width of the subform control (the white win¬ 
dow) is automatically set to equal the width of the 
subform. 



If you make changes to the size of your sub¬ 
form once the subform control is created, you 
may have to resize the subform control by 
clicking and dragging a corner handle. 


7.3.4 Linking forms and subforms 
manually 

If both the form and the subform are based on 
tables, and if relationships have been defined 
between the tables, Access normally has no problem 
determining which fields “link” the information on the 
main form with the information in the subform. How¬ 
ever, when the forms are built on queries, Access 
has no relationship information to rely on. As such, 
you have to specify the form/subform links manually. 


Since both the forms created in Section 7.3.3 were 
built on tables, Access could automatically deter¬ 
mine the relationship. 

• Verify the link between the form and the subform 
by examining the property sheet of the subform 
control, as shown in Figure 7.8. 

A The terminology “link child field” and “link 
. master field” is identical to “foreign key” and 
“primary key”. The main form is the parent 
(“one” side) and the subform is the child 
(“many” side). 

• View the resulting form. Notice that as you move 
from course to course, the number of sections 
shown in the subform changes (see Figure 7.9). 
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Tutorial exercises 


FIGURE 7.7: The drag-and-drop operation creates a subform control. 


B1 frmCoursesMain : Form 
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This is the name of the form to which 
the subform control is bound. 


+ Form Footer 
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Cre< 

Jits 

i 

Credits 

[Ac 

tivity 

_i_ 

Activit 


Jl 


The white area is a 
“subform control”. It is 
essentially a window 
through which the subform 
shows. 


© 


You may want to 
delete the label 
created with the 
subform window. To 
delete the label only, 
select it and press 
Delete. 


The form footer is pushed down when the subform control is created. You 
may move the footer to create more or less area at the bottom of the form. 
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7. Subforms 


Tutorial exercises 


FIGURE 7.8: Verify the link fields for the form/subform. 


B1 frmCoursesMain : Form 
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© 


Select the Sections subform 
control (the white window) and bring 
up its property sheet. 


Verify that Access has correctly 
determined the link fields. 


When there are more than one link 
fields (i.e., the foreign key is 
concatenated), separate the field 
names with a semicolon. In Access 
version 7.0, a builder is available to 
select the field names from a list. 
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Tutorial exercises 


FIGURE 7.9: A synchronized main form/subform. 
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Note that for COMM 
290, eight courses are 
listed in the subform. 


lb 


Click the “next 
record” navigation 
button on the main 
form to move to the 
next course. 


There are two sets of 
navigation buttons: 
one for the main form 
(bottom) and one for 
the subform (at the 
bottom of the 
subform window). 



For COMM 291, four 
sections are listed in 
the subform. 
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7. Subforms 


Tutorial exercises 


7.3.5 Non-synchronized forms 

In this section, you will delete the link fields shown in 
Figure 7.8 in order to explore some of the problems 
associated with non-synchronized forms. 

• Return to form design mode and delete the link 
fields (highlight the text and press the Delete 
key). 

• View the form. Note that all records in the Sec¬ 
tions table (not just those associated with a 
particular course) are shown. 

• Attempt to add a new section to COMM 290 as 
shown in Figure 7.10. 

• Re-establish the correct link fields and save the 
form. 

7.3.6 Aesthetic refinements 

In this section, you will modify the properties of sev¬ 
eral form objects (including the properties of the form 


itself) to make your form more attractive and easier 
to use. 

In Figure 7.11 , the basic form created in the previous 
sections is shown and a number of shortcomings are 
identified. 

7.3.6.1 Changing the form’s caption 

• Select the form as shown in Figure 7.12. 

• Change its Caption property to “Courses and 
Sections”. 

7.3.6.2 Eliminating unwanted scroll bars and 
navigation buttons 

Scroll bars and navigation buttons are also form- 
level properties. However, in this case, you need to 
modify the properties of the subform. 

• To quickly open the subform in design mode, 
double-click the subform control when viewing 
the main form in design mode (this takes some 
practice) 


| -^Homel H Previous | 13 of 19 | Next ► 1 


7. Subforms 


Tutorial exercises 


FIGURE 7.10: A non-synchronized main form/subform. 


Delete the li nk 
fields for the 
subform control 
and view the 
resulting form. 


lb 


Note that all 37 
sections show in 
the subform 
(moving to a 
different course 
has no effect). 

Add a new 
catalog number 
and click the 
record selector 
to try to save the 
new record. 
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Since the forms are not synchronized, the 
Dept Code and CrsNum fields of the Sections 
table are not automatically filled in by Access. 


r ok 


Help 
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7. Subforms 


Tutorial exercises 


FIGURE 7.11: A form/subform in need of some basic aesthetic refinements. 



The caption of the form shows the form’s name. 
A more attractive/descriptive caption is required. 


Since the subform control 
was automatically sized to 
fit the underlying form, a 
horizontal scroll bar is not 
necessary. 


The navigation buttons for 
the subform are too easily 
confused with the 
navigation buttons for the 
main form 
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7. Subforms 


Application to the assignment 


FIGURE 7.12: Select the entire form. 


Click on the square where the vertical 
and horizontal rulers meet in order to 
get the property sheet for the form. 
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• Bring up the property sheet for the form and scroll 
down to change its Scroll Bars and Navigation 
Button properties, as shown in Figure 7.13. 

The net result, as shown in Figure 7.14, is a more 

attractive, less cluttered form. 

7.4 Application to the assignment 

• Create a form and subform for your shipment 
and shipment Details information. You will 
use this form to record the details of shipments 

from your suppliers. 

Note that both forms should be based on queries: 

• the shipment form should be based on a “sort” 
query so that the most recent shipment always 
shows first; 

• the shipmentDetaiis form should be based 
on a join query so that validation information 
(such as the name of the product) is shown when 
a product number is entered. 
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7. Subforms 


Application to the assignment 


FIGURE 7.13: Change the scroll bars and 
navigation buttons of the subform. 


B1 sfrmSections : Form 
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Set the Scroll Bar 
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Only” and the Navigation 
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• Create a form/subform to show customer orders 
that have already been placed (such as the one 
you entered manually in Section 4.5). The top 
part of the form should contain information about 
the order plus some information about the cus¬ 
tomer; the subform should contain information 
about what was ordered and what was actually 
shipped. 



The form you created in the preceding step is 
used for viewing existing orders, not for add¬ 
ing new orders. To add new orders, the form 
must be more complex. For example, it has to 
show the quantity on hand and the back 
ordered quantity for each item so the user can 
decide how many to ship. You will create a 
form for order entry in the latter tutorials. 


• Set the Allow Additions and Allow Edits proper¬ 
ties of the “order viewing” form to No. This pre- 
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7. Subforms Application to the assignment 

FIGURE 7.14: A form without subform scroll bars or navigation buttons. 
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7. Subforms 


Application to the assignment 


vents the user from changing the details of an 
order that has already been invoiced or attempt¬ 
ing to use the form for order entry. 
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Access Tutorial 8: Combo Box Controls 


8.1 Introduction: What is a combo 
box? 

So far, the only kind of “control” you have used on 
your forms has been the text box. However, Access 
provides other controls (such as combo boxes, list 
boxes, check boxes, radio buttons, etc.) that can be 
used to improve the attractiveness and functionality 
of your forms. 

A combo box is list of values from which the user can 
select a single value. Not only does this save typing, 
it adds another means of enforcing referential integ¬ 
rity since the user can only pick values in the combo 
box. For example, a combo box for selecting course 
activities from a predefined list is shown in 
Figure 8.1 . 

Although advanced controls such as combo boxes 
and list boxes look and behave very differently than 
simple text boxes, their function is ultimately the 


FIGURE 8.1: A combo box for filling in the 

Activity field. 



same. For example, in Figure 8.1 , the combo box is 
bound to the Activity field. When an item in the 
combo box is selected, the string (e.g., “LEC”) is 
copied into the underlying field exactly as if you had 
typed the letters L-E-C into a text box. 


© Michael Brydon (brydon@unixg.ubc.ca) 
Last update: 25-Aug-1997 
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8. Combo Box Controls 


Learning objectives 



It is important to realize that combo boxes 
have no intrinsic search capability. Combo 
boxes change values—they do not automati¬ 
cally move to the record with the value you 
select. If you want to use a combo box for 
search, you have to program the procedure 
yourself (see Tutorial 15 for more details). 


8.2 Learning objectives 

□ How do I create a bound combo box? 

□ Can I create a combo box that displays values 
from a different table? 

□ How do I show additional information in a 
combo box? 

□ How do I prevent certain information from 
showing in the combo box? 

□ Can I change the order in which the items 
appear in a combo box? 


□ What is tab order? How do I change it so that 
the cursor moves in the correct order? 

□ Should I put a combo box on a key field? 

8.3 Tutorial exercises 

• Open your frmCourses form in design mode. 

• Ensure the toolbox and field list are visible (recall 
Figure 6.3). 

8.3.1 Creating a bound combo box 

Although Access has a wizard that simplifies the pro¬ 
cess of creating combo boxes, you will start by build¬ 
ing a simple combo box (similar to that shown in 
Figure 8.1) with the wizard turned off. This will give 
you a better appreciation for what the wizard does 
and provide you with the skills to make refinements 
to wizard-created controls. 

• Delete the existing Activity text box by select¬ 
ing it and pressing the Delete key. 
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8. Combo Box Controls 


Tutorial exercises 


• The wizard toggle button ( \|) in the toolbox 
allows you to turn wizard support on and off. 
Ensure the button is out (wizards are turned off). 

• Click on the combo box tool ( ml). The cursor 
turns into a small combo box. 

• With the combo box tool selected, drag the 
Activity field from the field list to the desired 
location on the form’s detail section, as shown in 
Figure 8.2. 

The process of selecting a tool from the toolbox, and 
then using the tool to drag a field from the field list 
ensures that the control you create (text box, combo 
box, etc.) is bound to a field in the underlying table or 
query. 

A If you forget to drag the field in from the field 
. list, you will create an unbound combo box, as 
shown in Figure 8.3. If you accidently create 


an unbound combo box, the easiest thing to 
do is to delete it and try again. 

FIGURE 8.3: An unbound combo box (not what 

you want). 
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Since the control 
is unbound, no 
field name 
shows and the 
label is generic. 
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Tutorial exercises 


FIGURE 8.2: Create a bound combo box. 


IS frmCourses : Form 
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d 



Drag the Activity field on to the detail area. If you 
have done this correctly, the name of the underlying 
field should show in the combo box and the label 
should take the value of the field’s caption 


Ensure the wizard button is not 
depressed. 


Click on the combo box button to 
activate the combo box tool. 


Select the Activity field from 
the field list. 


I ome ]E Previous] 4 of 23 | Next ► ] 








































































































































































8. Combo Box Controls 


Tutorial exercises 


8.3.2 Filling in the combo box properties 

In this section, you will tell Access what you want to 
appear in the rows of new combo box. 

• Switch to form view and test the combo box. 

At this point, the combo box does not show any list 
items because we have not specified what the list 
items should be. There are three methods of specify¬ 
ing what shows up in the combo box list: 

1. enter a list of values into the combo box’s Row 
Source property; 

2. tell Access to get the value from an existing table 
or query; 

3. tell Access to use the names of fields in an exist¬ 
ing table (you will not use this approach). 

Although the second method is the most powerful 
and flexible, you will start with the first. 

• Bring up the property sheet for the Activity 
combo box. 


• Change the Row Source Type property to value 
List as shown in Figure 8.4. This tells Access to 
expect a list of values in its Row Source property. 

FIGURE 8.4: Set the Row Source Type property. 
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8. Combo Box Controls 


Tutorial exercises 


• Enter the following into the Row Source property: 

LAB;LEC;TUT 

• Set the Limit To List property to Yes. 



If the Limit To List property is set to No, the 
user can ignore the choices in the combo box 
and simply type in a value (e.g., “SEM”). In 
this particular situation, you want to limit the 
user to the three choices given. 


• Switch to form view and experiment with the 
combo box. 



Notice that the combo box has some useful 
built-in features. For example, if you choose 
to type values rather than select them with a 
mouse, the combo box anticipates your 
choice based on the letters you type. Thus, to 
select “TUT”, you need only type “T”. 


8.3.3 A combo box based on another 
table or query 

An obvious limitation of the value-list method of cre¬ 
ating combo boxes is that it is impossible to change 
or update the items that appear in the list without 
knowing about the Row Source property. 

A more elegant and flexible method of populating the 
rows of a combo box is to have Access look up the 
values from an existing table or query. Although the 
basic process of setting the combo box properties 
remains the same, it is more efficient to rely on the 
wizard when building this type of combo box. 

Before you can continue, you need a table that con¬ 
tains appropriate values for course activities. 

• Switch to the database window and create a new 
table called Activities. 

• The table should consist of two fields: one called 
Activity and the other called Descript, as 
shown in Figure 8.5. 
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8. Combo Box Controls 


Tutorial exercises 


FIGURE 8.5: Create a table containing course 

activities. 
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• Populate the table with the same values used in 
Section 8.3.2. 

The result is a table containing all the possible 
course activities and a short description to explain 
the meaning of the three-letter codes. You can now 
return to creating a combo box based on these val¬ 
ues. 

• Delete the existing Activity combo box. 

• Ensure the wizard button (ps) in the toolbox is 
depressed (wizards are activated). 

• Repeat the steps for creating a bound combo box 
(i.e., select the combo box tool and drag the 
Activity field from the field list on to the detail 
section). As shown in Figure 8.6, this activates 
the combo box wizard. 

The wizard asks you to specify a number of things 
about the combo box: 


1. the table (or query) from which the combo box 
values are going to be taken; 
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FIGURE 8.6: Create a combo box using the combo box wizard. 

Create a bound 
combo box. 
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8. Combo Box Controls 


Tutorial exercises 


2. the field (or fields) that you would like to show up 
as columns in the in the combo box; 

3. the width of the field(s) in the combo box (see 
Figure 8.7); 

4. the column from the combo box (if more than one 
field is showing) that is inserted into the underly¬ 
ing field; and, 

5. the label attached to the field (see Figure 8.8). 

When you are done, the combo box should look sim¬ 
ilar to that shown in Figure 8.1 . However, updating or 
changing the values in the combo box is much easier 
when the combo box is based on a table. 

• Add “SEM” (Seminar) to the Activities table. 

• Return to the form, click on the Activity combo 
box, and press F9 to requery the combo box. 

• Verify that “SEM” shows up in combo box. 



Access creates the rows in a combo box 
when the form is opened. If the values in the 


source table or query change while the form is 
open these changes are not automatically 
reflected in the combo box rows. As a conse¬ 
quence, you have to either (a) close and re¬ 
open the form, or (b) requery the form. 
Although you can automate the requery pro¬ 
cess, we will rely on the F9 key for the time 
being. 

8.3.3.1 Showing more than one field in the 
combo box 

One problem the combo boxes created so far is that 
they are not of much use to a user who is not familiar 
with the abbreviations “TUT”, “SEM”, and so on. In 
this section, you will use the Descript field of the 
Activities table to make the combo box more 
readable, as shown in Figure 8.9. 

• Delete the existing combo box and start again. 
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Combo Box Wizard 



Which table or query should provide the values for your combo box? 
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FIGURE 8.7: Fill in the combo box 
wizard dialog sheets. 
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Which fields contain the values you want included in your combo 
box? 


The fields you select become columns in your combo box. 

The combo box can show 
more than one field. Select 
only Activity for now. 


Available Fields: 


Selected Fields: 






Descript 


r 


Activity 




Combo Box Wizard 


How wide would you like the columns in your combo box? 


To adjust the width of a column, drag its right edge to the width you want or double-click the right 
edge of the column heading to get the best fit. 


Use the column selector (the grey bar 
at the top of the column) to resize the 
column to the desired width. 
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8. Combo Box Controls 


Tutorial exercises 


FIGURE 8.8: Fill in the combo box wizard dialog sheets (continued). 


Combo Box Wizard 


^9 The combo box is already bound 
to the Activity field, this step 
is automatically filled in for you. 


m 

!■!!■!!■! 

!■!!■!!■! 

KHK 

!■!!■!!■! 

!■!!■!!■!!■! 


Microsoft Access can store the selected value from your combo box 
in your database., or remember the value so you can use it later to 
perform a task. 



Remember the value for later use. 


iStore that value in this field: lActivity 


3 


!■!!■!!■! !■!!■!!■! !■!!■!!■! KKK 

m 

!■!!■!!■! !■!!■!!■! !■!!■!!■! !■!!■!!■! 


Because the combo box is bound, 
the Activity field’s caption is 
provided as a default label. 
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8. Combo Box Controls 


Tutorial exercises 


FIGURE 8.9: A combo box that shows two fields 
from the source table or query. 



• Fill in the wizard dialog sheets as in Section 8.3.3 
but make the changes shown in Figure 8.10. 

• Verify that your combo box resembles Figure 8.9. 

8.3.3.2 Hiding the key field 

Assume for a moment that you, as a developer, do 
not want users to even see the three-letter abbrevia¬ 


tions and want them to select a course activity value 
based solely on the Descript field. 

In such a case, you could include only the 
Descript column in the combo box. However, this 
would not work because the Activity field of the 
Courses table expects a three-letter abbreviation. 
As such, the combo box would generate an error 
when it tried to stuff a long description into the rela¬ 
tively short field to which it is bound. 

In this section, you will create a combo box identical 
to that shown in Figure 8.9 except that the key col¬ 
umn (Activity) will be hidden from view. Despite 
its invisibility, however, the Activity column will 
still be bound to the Activity field of the underly¬ 
ing table and thus the combo box will work as it 
should. 

• Delete the existing combo box and start again 
using the combo box wizard. 
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8. Combo Box Controls 


Tutorial exercises 


Combo Box Wizard 


kkk kkk m kkk 
kkk kkk kkk kkk 

KKK KKK KKK KKK 


Which fields contain the values you want included in your combo 
box? 

The fields you select become columns in your combo box. 


FIGURE 8.10: Use the wizard to 
add more than one field to the 
combo box. 


KKKKKK 


3 


Available Fields: 


Selected Fields: 



, 

Activity ^ ^ 

> 

Descript 

„ 1 

1 


Combo Box Wizard 


How wide would you like the colurmjS'+rfyour combo box? 


Bring both fields from the 
Activities table into the combo box. 


Uncheck the “hide key” box and 
resize the columns appropriately. 
Note that Access version 2.0 does 
not have the “hide key” feature 



To adjust the widthofja<^umn, drag its right edge to the width you want or double-click th^ril 
edge of the^plyrrTffheading to get the best fit. 


Combo Box Wizard 


I - Hide key co I u mi n (re co rn mi e n d e d) 


Select the column that provides 
the value of interest (in this case, 
Activity). 



Activity 

Descript 

ii>~; 

m . 

Lab 


LEC 

Lecture 


TUT 

Tutorial ! 


KKKKKKK 


“3 

KKKK 

KKKK 

KKKK 

KKK 

KKK 

KKK 

KKKK 

KKK 

KKKK 



When you select a row in the combo box. you can store a value from 
that row in your database, or you can use the value later to perform 
an acti o n. Ch o o s eafieldth at unique ly i d e ntifi e s th e row. 

Available Fields: 


Descript 
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8. Combo Box Controls 

• Include both the Activity and Descript fields 
in the combo box. 

• Resize the Activity column as shown in 
Figure 8.11 . Note that users of version 7.0 can 
simply leave the “hide key” box checked—the 
result is the same. 

• Ensure that the Input Mask property for the 
combo box (which is inherited from the field’s 
Input Mask property) is blank. 

• Verify that the resulting combo box resembles 
that shown in Figure 8.12. 

A Combo boxes with hidden keys can be con- 
. fusing. The important thing to remember is 
that even though the description (e.g., “Lec¬ 
ture”) now shows in the combo box, what is 
really stored in the underlying field is the hid¬ 
den key (e.g., “LEC”). 


Tutorial exercises 


FIGURE 8.12: A combo box with a hidden key. 



8.3.3.3 Changing the order of items in the 
combo box 

A combo box based on a table shows the records in 
one of two ways: 

1. If the table does not have a primary key, the 
records are shown in their natural order (that is, 
in the order they were added to the database). 
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8. Combo Box Controls 


Tutorial exercises 


FIGURE 8.11: Resize the columns to hide the key. 


© 


Click on the right side of 
the column selector and 
drag the edge of the 
Activity column to the 
far left (i.e., make its width 
zero) 


Hiding the key is such a 
common operation that 
Access version 7.0 includes 
the “hide key” check box. 


Combo Box Wizard 


How wide would you like the columns 

To adjust the width of a column, drag 
idge of the column heading to get thi 


Combo Box Wizard 


How wide would you like the 


Combo Box Wizard 



r - 

Activity 

Descript^ 

IS 

IlabI • 

Lab 


LEG 

Lecture 


SEM 

Seminar 


TUT 

Tutorial 



_ H o'w wi d e wo u I d yo u I i ke th e co 1 1 . 

To adj u st th e wi dth of a co I un 

edge ofthe column heading- jo adjust the width of acolumn, c 

edge ofthe column heading to gi 


I - Hide key column (recommen 


column (recomr 


Descript 


Lab 


LE Lecture 


SE Seminar 


TL 


Tutorial 



Descript 


EPS 


Lecture 


Seminar 


Tutorial | 


— 

— 


Ci: 
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8. Combo Box Controls 

2. If the table does have a primary key, then the 
records are sorted in ascending order according 
to the key. 

It may be, however, that you want a different order 
within the rows of the combo box. To achieve this, 
you can do one of two thing: 

1. Create a stand-alone query (in which the sort 
order is specified) and use this query as the 
source for the combo box. 

2. Modify the “ad hoc” query within the Row Source 
property of the combo box. 

If you intend to make several major changes to the 
basic information in the underlying table (e.g., joins, 
calculated fields), it is usually better to create a 
stand-alone query. In this way, the same query can 
be used by many combo boxes. 


Tutorial exercises 

If the changes are quite minor (for instance, sorting 
the records in a different order), you may prefer to 
modify the Row Source property. 

In Section 8.3.2, you set the Row Source property to 
equal a list of values. When the combo box is based 
on values from a table or a query, however, the Row 
Source is an SQL statement (recall Tutorial 5) rather 
than a list of values. You can either edit the SQL 
statement directly or invoke the QBE editor. 

In this section, you will order the items in you combo 
box according to the length of the Descript field 
(this is done merely for illustrative purposes). 

• Bring up the property sheet for the Activity 
combo box. 

• Put the cursor in the Row Source property. As 
shown in Figure 8.13, a builder button ( - ) 
appears. 

• Press the builder button to enter the “SQL 
builder” (i.e., the QBE editor). 
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8. Combo Box Controls 


Tutorial exercises 


FIGURE 8.13: Invoke the builder for the Row 

Source property. 


I i.-.+ IT ! 1 



if Combo Box: Activity 


Format 

Name. Activity 

Control Source 
Format 

Decimal Places . 

Input Mask 
RJw Source Type 
Row Source 
l-L.!.: iVi n Count 
Column Heads 
Column Widths 
Bound Column 




Click the builder button to 
bring up the QBE editor. 
Alternatively, you can edit 
the SQL statement directly. 


• Create a calculated field called DescLength 
using the following expression: 

DescLength: Len([Descript]) 


(ten () is a built-in function that returns the 
length of a string of characters). 

• Sort on DescLength in descending order. 

• Switch to datasheet view to ensure the query is 
working as it should. 

• Ensure the Show box for the field is unchecked, 
as shown in Figure 8.14. 

• Instead of saving the query in the normal way, 
simply close the QBE box using the close button 

(id). 

A If you save the query, it will be added to your 

»\ collection of saved queries (the ones that are 
displayed in the database window). However, 
if you simply close the QBE window, the Row 
Source property will be updated and no new 
database object will be created. 
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Tutorial exercises 


FIGURE 8.14: Use the QBE editor to modify the 

Row Source property. 


i 1 SQL Statement: Query Builder 


Activities 


Activity 

Descript 


Add a calculated field 
called DescLength. 


ijj 


Field 

Table 

Sort 

Show 

Criteria 

or 


Activity 

Descript 

DescLenqth: Le n (I'D e script]) 

Activities 

Activities 




Descendinq 







■ ggjgggg 

Sort on the 


Uncheck the - 

calculated field. 


Show box 


8.3.4 Changing a form’s tab order 

A form’s tab order determines the order in which the 
objects on a form are visited when the Tab or Enter 
(or Return) keys are pressed. Access sets the tab 
order based on the order in which objects are added 
to the form. As a result, when you delete a text box 
and replace it with a combo box or some other con¬ 
trol, the new control becomes the last item in the tab 
order regardless of its position on the form. 

To illustrate the problem, you are going to create a 
combo box for the Deptcode field. 

• Delete the Deptcode text box and replace it with 
a combo box based on the Departments table. 

• Switch to form view. Notice that the focus starts 
off in the crsNum field instead of the Deptcode 
field. 

• Press tab to move from field to field. Notice that 
after Deptcode is left, the focus returns to the 
CrsNum field of the next record. 


I ome 1 Previous] 18 of 23 | Next ► ] 














































































































8. Combo Box Controls 


Discussion 


• To fix the problem, return to form design mode 
and select View> Tab Order from the main 
menu. 

A in Access version 2.0, the menu structure is 
. slightly different. As such, you must select 
Edit > Tab Order. 

• Perform the steps in Figure 8.15 to move Dept- 
Code to the top of the tab order. 

8.4 Discussion 

8.4.1 Why you should never use a 

combo box for a non-concatenated 
key. 

A mistake often made once new users learn how to 
make combo boxes is to put a combo box on every¬ 
thing. There are certain situations, however, in which 
the use of a combo box is simply incorrect. 


For example, it never makes sense to put a combo 
box on a non-concatenated primary key. To illustrate 
this, consider the Departments form shown in 
Figure 8.16. On this form, the Deptcode text box 
has been replaced with a combo box that draws its 
values from the Departments table. 

FIGURE 8.16: A combo box bound to a key field. 


BH Departments 

► 

Department code 

COMM 

T 


Department name 

BSKVV 

Basket Weaving 


COMM 

Commerce and Business Admin 



CP.WR 

Creative Writing 


Building 

EDUC 

Education 



ENGL 

Enalish 

Record: M | ^ | 

MATH 

l_ 1 1 LH 1 1 w 1 1 

Math 


MUSC 

Music 


This combo box appears to work. However, if you 
think about it, it makes no sense: The form in 
Figure 8.16 is a window on the Departments table. 
As such, when the Deptcode combo box is used, 
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Discussion 


FIGURE 8.15: Adjust the tab order of fields on a form. 


Drag the record 
selector to the 
desired position in 
the list. 


Tab Order 


HSection 

HeliiS 

Detail 

C ^pilfe^pr 

Click to select a row, or 
click and drag to select 
multiple rows. Drag 
selected row(s) to move 
them to desired tab order. 



Custom Order: 


CrsNum 
Title 
Cr edits 
Aativft 


DeptCode 


fab Order 


OK 


Cancel 


© 


For forms in which the fields are arranged 
in a single column from top to bottom 
(such as this one), you can press Auto 
Order to order them automatically. 


Section 

O ft|rr|i®lii(|lr 
<* Detail 
C pggp jjjj fflM 


Click to select a row, or 
click and drag to select 
multiple rows. Drag 
selected row(s) to move 
them to desired tab order. 


Click on the record 
selector of the field 
you wish to move. 


Custom Order: 


DeptCode 


CrsNum 

Title 

Credits 

Activity 


"OK“ 


"Cancel - 


Auto Order 
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8. Combo Box Controls 


Discussion 


one of two things can occur depending on whether a 
new record is being created or an existing record is 
being edited: 

1 . A new record is being created — If a new 

record is being created (i.e., a new department is 
being added to the information system), a unique 
value of Deptcode must be created to distin¬ 
guish the new department from the existing 
departments. However, the combo box only 
shows Deptcode values of existing depart¬ 
ments. If the Limit To List property is set to Yes, 
then the combo box prevents the user from enter¬ 
ing a valid Deptcode value. 

2. An existing record is being edited — It is 
important to remember that a combo box has no 
intrinsic search capability. As such, selecting 
“CPSC” in the Deptcode combo box does not 
result in a jump to the record with “CPSC” as its 
key value. Rather, selecting “CPSC” from the 


combo box is identical to typing “CPSC” over 
whatever is currently in the Deptcode field. This 
causes all sorts of problems; the most obvious of 
these is that by overwriting an existing value of 
Deptcode, a “duplicate value in index, primary 
key, or relationship” error is generated (there is 
already a department with “CPSC” as its Dept¬ 
code). 

Note that a combo box may make sense when the 
key is concatenated. An example of this is the 
Deptcode combo box you created in Section 8.3.4. 

8.4.2 Controls and widgets 

Predefined controls are becoming increasingly popu¬ 
lar in software development. Although Microsoft 
includes several predefined controls with Access 
(such as combo boxes, check boxes, radio buttons, 
etc.), a large number of more compex or specialized 
controls are available from Microsoft and other ven- 
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8. Combo Box Controls 

dors. In addition, you can write your own custom 
controls using a language like Visual C++ or Visual 
Basic and use them in many different forms and 
applications. 

An example of a more complex control is the calen¬ 
dar control shown in Figure 8.17. A calendar control 
can be added to a form to make the entry of dates 
easier for the user. Microsoft calls such components 
“ActiveX controls” (formerly known as “OLE con¬ 
trols”). Non-microsoft vendors provide similar com¬ 
ponents but use different names, such as “widgets”. 

There are two main advantages of using controls. 
First, they cut down on the time it takes to develop 
an application since the controls are predefined and 
pre-tested. Second, they are standardized so that 
users encounter the same basic behavior in all appli¬ 
cations. 


Application to the assignment 

8.5 Application to the assignment 

There are a number of forms in your assignment that 
can be greatly enhanced by combo boxes. 

• Create a combo box on your order form to allow 
the user to select customers by name rather than 
CustiD. Since your CustiD value is a counter, it 
has no significance beyond its use as a primary 
key. Generally, such keys should be hidden from 
view. 

• Create a combo box in your order details subform 
to allow the user to select products. Since the 
Product id values are used by both you and 
your customers, they have some significance 
beyond the information system. As such, Pro¬ 
duct id should be visible in all combo boxes. In 
addition, the items in the product list should be 
sorted by Product id. This makes it easier to 
select a product by typing the first few numbers. 

• Create combo boxes on other forms as required. 
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Application to the assignment 


FIGURE 8.17: A calendar control on a form. 


Hi Forml : Form 


© 


# Detail 


August 1997 [ August ■H 1997 


Sun 

Mon 

Tue 

Wed 

Thu 

F 

27 

28 

29 

30 

31 


3 

4 

5 

6 

s 

8 

1 10 

11 

12 ^ 

4^ 

14 

15 

17 

18 ^ 

& 

20 

21 

22 

4 

25 

26 

27 

28 

29 

31 

1 

2 

3 

4 

5 

h__-_ 


Sat 


The calendar control can be bound 
to date/time fields, thereby making 
it easier for users to enter dates. 


I# Custom Control: OLEControlO 


Format 


On Updated 
On Enter... 

On Exit 
On Got Focus N 
On Lost Focus 



Like other objects in Access, controls have 
properties and events that determine the 
appearance and behavior of the control. 
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Access Tutorial 9: Advanced Forms 


9.1 Introduction: Using calculated 
controls on forms 

It is often useful to show summary information from 
the subform on the main form. The classic example 
of this is showing the subtotal from a list of order 
details on the main order form. 

In this tutorial, you are going to explore one means 
of implementing this feature using calculated con¬ 
trols. A calculated control is an unbound control for 
which the Control Source property is set to an 
expression that Access can evaluate. 

Clearly, calculated controls have a great deal in com¬ 
mon with the calculated query fields you created in 
Section 4.3.3. Although there are no hard-and-fast 
rules that dictate when to use a one over the other, 
pushing your calculations to the lowest level (i.e., 
performing calculations in the query) is usually the 


© Michael Brydon (brydon@unixg.ubc.ca) 
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9. Advanced Forms 

half year]. Recall that you have already imple¬ 
mented this feature in Section 4.3.3.2 using a calcu¬ 
lated query field. 

• Perform the steps shown in Figure 9.1 to create 
an unbound text box on your fmrCoursesMain 
form. 

• Set the Control Source property of the text box 
using the syntax: 

= <expression> 

In this case, the expression should be an “imme¬ 
diate if” function (see Section 4.3.3. 2). 

A By default, Access interprets text in the Con- 
. trol Source property field as the name of a 
variable (i.e., the name of a field or another 
control). As such, you must remember to 
include the equals sign when setting this 
property. 


best course of action. However, as you will see in the 
context of subtotals, this is not always possible. 

9.2 Learning objectives 

□ How do I create a calculated text box? 

□ What is the expression builder? When is it 
used? 

□ Where can put an intermediate result of a 
calculation on a form so that it does not 
show? 

9.3 Tutorial exercises 

9.3.1 Creating calculated controls on 
forms 

In this section, you are going to create a simple cal¬ 
culated text box to translate the credits field into a 
dichotomous text variable [full year, 
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Tutorial exercises 

• Test your form. Note that you are prevented from 
editing the calculated field. If, however, you 
change the value of credits, the value of txt- 
CourseLength changes accordingly when you 
leave the credits field. 

9.3.2 Showing a total on the main form 

In this section, you will create a calculated text box 
that displays the number of sections associated 
with each course. The primary motivation for this 
exercise is to illustrate some of the limitations of cal¬ 
culated controls (as they are implemented in Access) 
and to provide an opportunity to explore an interest¬ 
ing work-around. 

• Create a text box call txtNumSections on the 
main form as shown in Figure 9.2. 

The logical next step is to set the Control Source of 
the field to an expression that includes the count () 
function. However, Access has a limitation in this 
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Tutorial exercises 


il frmCoursesMain : Form 


FIGURE 9.1: Create an unbound text box on your main form. 

lb 


Make some room by dragging the 
Credits text box to the left. 
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* Form Header 
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Department c 
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CrsNun 
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13 frmCoursesMain : Form 


'extl 2: 




ifjnbourid 
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Act i v i ty h Activity 
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t 

□ 

_i 

r~ 


pH 

IP! 


a 

m 


Select the text box 
tool from the 
toolbox and click on 
an appropriate space 
in the detail area. 

Adjust the tab order 
of the fields as 
necessary. 

-1^1*1 


Secions 


sfrmSectic 


i ' 1 ' i ' 2 ' i 


'3' 


l '4' ' '5' l '6' l 


'L 


8 ' l 


1 ' 10 ' 1 ' y ' 1 ' 12 ' 1 ' 11 


+ Form Header 


^ Detail 


a 

Hr Fs 


Depa rtment c 3de DeptCc 


Get fse - tmr ibef- CrsNun 


I 


itle 


I 


Credits 


Credits 




Course length 


I 


Inbound 


T 


A 


□ 


abl 


i 


1 ext Box: txtCou rse Length 


m 


Title 


Sec 

ions 









Edit the label and give the text box a meaningful name 
(e.g., txtCourseLength). The txt prefix is used 
here to indicate an unbound text box. 


Format 
Name. 
Control Sj 
Paf 


Data Event Other 


All 


ixtCou rse Length 


:e ... . 


dll 


Decimal Places .... Auto 

Input Mask. 

Default Value ... 
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Tutorial exercises 


FIGURE 9.2: Create an unbound text box to show the number of sections 

associated with each course. 


Add an unbound text box called txtNumSections. 
Since it is currently bound to nothing, it is blank. 


Bl Courses and Sections 


Department code 
Course number 
Title 

Sections 


COMM 


290 



Co u rs e I e n gth h alt ye ar 


Number of sections: 


Activity 


LEC 


Introduction to Quantative Decision Making 


Catalog 

Num 

Section 

Session 

Term 

Meeting 

days 

Meeting 

time 

Building 

Room 

3 

► 

AAA)) 

001 

94W 

1 

M W 

830-1 000 

ANGU 

413 



57455 

002 

94W 

1 

W F 

830-1 000 

ANGU 

415 



48516 

003 

94W 

1 

W F 

1 030-1 200 

ANGU 

415 



71845 

004 

94W 

1 

M W 

1 000-1 130 

ANGU 

413 



69495 

005 

94W 

1 

MF 

1 300-1 430 

ANGU 

415 



34134 

006 

94W 

1 

MW 

1 300-1 430 

ANGU 

413 



What you want 
is a means of 
counting the 
records in the 
subform and 
displaying the 
count in the 
new text box. 
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9. Advanced Forms 


Tutorial exercises 


regard: you cannot use an aggregate function 
(Sum (), Avg (), Count (), etc.) on a main form that 
refers to a field in a subform. As a consequence, you 
have to break the calculation into two steps: 

1. use the aggregate function to create a calculated 
text box on the subform (i.e., a “dummy” field to 
hold an intermediate result); 

2. create a calculated control on the main form that 
references the dummy text box created in the first 
step. 

A lt is important that you realize that this proce- 
. dure does not involve any immutable, funda¬ 
mental information systems knowledge. 
Rather, it is merely an example of the type of 
work-around (hack, kludge, etc.) that is rou¬ 
tinely used when using a tool like Access to 
create a custom application. 


9.3.2.1 Calculating the aggregate function on 
the subform 

• Create an unbound text box on the subform as 
shown in Figure 9.3. 

• Save the subform but do not close it. 

• Return to the main form and set the Control 
Source of txtNumSections to equal the value 
of txtNumSectionsOnSub. Since the naming 
conventions for objects on forms and subforms 
can be tricky, use the expression builder (as 
shown in Figure 9.4) to build the name for you. 

The expression builder organizes all the elements of 
the database environment into a hierarchical struc¬ 
ture. You build an expression by “drilling down” to the 
element you need and double-clicking to copy its 
name into the text area. 



The expression builder takes some practice. 
One problem is that it is easy to double-click 
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Tutorial exercises 


FIGURE 9.3: Perform the count on the subform. 
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9. Advanced Forms 


Tutorial exercises 


FIGURE 9.4: Use the builder to drill down to the calculated control on the subform. 


© 


Note that when the main 
form and the subform are 
both open, the subform 
appears twice in the builder: 
once as a “stand-alone” 
form (under “Loaded 
Forms”) and once as a 
component of the main form 
(press the + sign on the 
frmCoursesMain 
folder). You want to use the 
latter (you will never 
access the subform in stand¬ 
alone mode). 
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Invoke the builder from the 
Control Source property and drill 
down to the calculated control you 
just created on the subform. 
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9. Advanced Forms 


Tutorial exercises 


on the wrong thing. Another problem is that 
Access attempts to guide you by inserting 
«Expr» place-holders all over the place. The 
solution to both problems is to click on the text 
window and make liberal use of the Delete 
key. 

A The point made about “stand-alone” and 
. “component” subforms in Figure 9.4 is 
extremely important. The reason you use the 
sfrm prefix is so you know that the form is 
designed to be a component of another form. 
If you select the stand-alone version the form 
in the builder, the name created by the builder 
will be incorrect and an error will result. 

• Close the subform (in version 7.0 and 8.0, the 
main form and subform cannot be open at the 
same time). 


• Test the form. The value Of txtNumSections 
and txtNumSectionsOnSub should be identi¬ 
cal, as shown in Figure 9.5. 

FIGURE 9.5: The number of sections on the main 

form. 


gi Courses and Sections 


Department code 
Course number 
Title 

Sections 
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© 


GJ 


Tl TTTkTwT 

The “dummy” text box is visible in 
the subform. Although you will 
eventually hide it, it is useful to 
display it until you know both steps 
of the calculation are working 
properly. 
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9. Advanced Forms 


Discussion 


9.3.2.2 Hiding the text box on the subform 

The obvious problem in Figure 9.5 is that the dummy 
text box shows on the subform. There are at least 
two ways to get around this: one is to set the Visible 
property of the text box to No; a slightly more elegant 
approach is to use the page header or page footer 
to hide the text box. 

The page header and footer are areas on the form 
that only show when the form is printed. Since you 
will never print a form (reports are used for printed 
material), these areas can be used to hide intermedi¬ 
ate results, etc. 

• In design mode, select View > Page Header/ 
Footer from the menu. 

A in version 2.0, the menu structure is slightly 
. different. As such, you must select Format > 
Page Header/Footer. 


• Drag (or cut and paste) txtNumSectionsOn- 
Sub from the form header to the page header, as 
shown in Figure 9.6. 

• Test the result. 

9.4 Discussion 

In Section 4.3.3.2 and Section 9.3.1 , you accom¬ 
plished the same thing (showing half year or 
full year) using different techniques. The advan¬ 
tage of implementing this as a calculated query field 
is that you can use this field repeatedly in other 
forms. On the other hand, if you do the transforma¬ 
tion on the form, you have to repeat the calculation 
on every form that requires the calculated field. 

In the case of the aggregate function, the situation is 
slightly different. Although you can use the totals 
feature of QBE (see on-line help) to count the num¬ 
ber of sections for a particular course within a query, 
the resulting recordset is non-updatable (and hence 

- I '^Homel H Previous! 9 of 11 | Next ► 1 


9. Advanced Forms 


Discussion 


FIGURE 9.6: Hide the intermediate result in the page header. 
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9. Advanced Forms 


Application to the assignment 


not much use for editing course names, etc.). As a 
result, you are forced to do the calculation on the 
form rather than in the query. 

9.5 Application to the assignment 

To show the subtotal, tax, and grand total on your 
order form, you use the same techniques illustrated 
here. The only difference is that you use the Sum () 
function instead of the Count () function to get the 
subtotal for the order. 

• Create a dummy field on your OrderDetaiis 
subform to calculate the subtotal for the order. 

• Calculate the tax (G.S.T. only for wholesale) and 
grand total on the main form (traditionally, this 
information is located near the bottom of the 
form—but not in the form footer). 
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Access Tutorial 10: Parameter Queries 


The last few tutorials have been primarily concerned 
with interface issues. In the remaining tutorials, the 
focus shifts to transaction processing. 

10.1 Introduction: Dynamic queries 
using parameters 

A parameter query is a query in which the criteria 
for selecting records are determined when the query 
is executed rather than when the query is designed. 

For example, recall the select query shown in 
Figure 4.6. In this query, the results set is limited to 
records that satisfy the criterion Deptcode = 
"comm". If you wanted a different set of results, you 
would have to edit the query (e.g., change the crite¬ 
rion to "cpsc") and rerun the query. 

However, if a variable (parameter) is used for the cri¬ 
terion, Access will prompt the user for the value of 
the variable before executing the query. The net 

© Michael Brydon (brydon@unixg.ubc.ca) 
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10. Parameter Queries 

10.3 Tutorial exercises 

10.3.1 Simple parameter queries 

• If you do not already have a qryCourses query 
like the one shown in Figure 4.6, create one now 
and save it under the name pqrycourses. 

• Replace the literal string in the criteria row 
("comm") with a variable ( [x]). 

A By default, Access expects criteria to be literal 
. strings of text. As a result, it automatically 
adds quotation marks to text entered in the 
criteria row. To get around this, place your 
parameter names inside of square brackets. 

• Execute the query as shown in Figure 10.1. 

When Access encounters a variable (i.e., something 
that is not a literal string) during execution, it 


result is that parameters can be used to create 
extremely flexible queries. 

When the concepts from this tutorial are combined 
with action queries (Tutorial 11) and triggers 
(Tutorial 13), you will have a the skills required to 
create a simple transaction processing system with¬ 
out writing a line of programming code. 

10.2 Learning objectives 

□ What is a parameter query? How do I create 
one? 

□ How do I prompt the user to enter parameter 
values? 

□ How do I create a query whose results 
depend on a value on a form? 
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Tutorial exercises 

attempts to bind the variable to some value. To do 

this, it performs the following tests: 

1. First, Access checks whether the variable is the 
name of a field or a calculated field in the query. If 
it is, the variable is bound to the current value of 
the field. For example, if the parameter is named 
[Deptcode] , Access replaces it with the current 
value of the Deptcode field. Since x is not the 
name of a field or a calculated field in this particu¬ 
lar query, this test fails. 

2. Second, Access attempts to resolve the parame¬ 
ter as a reference to something within the current 
environment (e.g., the value on an open form). 
Since there is nothing called x in the current envi¬ 
ronment, this test fails. 

3. As a last resort, Access asks the user for the 
value of the parameter via the “Enter Parameter 
Value” dialog box. 
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10. Parameter Queries 


Tutorial exercises 


FIGURE 10.1: Convert a select query into a parameter query. 
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10. Parameter Queries 



Note that the spelling mistakes discussed in 
Section 4.3.4 are processed by Access as 
parameters. 


10.3.2 Using parameters to generate 
prompts 

Since the name of the parameter can be anything 
(as long as it is enclosed in square brackets), you 
can exploit this feature to create quick and easy dia¬ 
log boxes. 

• Change the name of your DeptCode parameter 
from [X] to [Courses for which depart¬ 
ment ? ]. 

• Run the query, as shown in Figure 10.2. 


10.3.3 Values on forms as parameters 

A common requirement is to use the value on a form 
to influence the outcome of a query. For instance, if 
the user is viewing information about departments, it 


Tutorial exercises 

may be useful to be able to generate a list of courses 
offered by the department currently being viewed. 
Although you could use a creatively-named parame¬ 
ter to invoke the “Enter Parameter Value” dialog, this 
requires the user to type in the value of DeptCode. 

A more elegant approach is to have Access pull the 
value of a parameter directly from the open form. 
This exploits the second step in the operation of a 
parameter query (Access will attempt to resolve a 
parameter with the value of an object within the cur¬ 
rent environment). The basic idea is shown in 
Figure 10.3. 

The key to making this work is to provide a parame¬ 
ter name that correctly references the form object in 
which you are interested. In order to avoid having to 
remember the complex naming syntax for objects on 
forms, you can invoke the expression builder to 
select the correct name from the hierarchy of data¬ 
base objects. 
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10. Parameter Queries 


Tutorial exercises 


FIGURE 10.2: Select a parameter name that generates a useful prompt. 


as 1 pqryCourses : Select Query 


Courses 


DeptCode 

CrsNum 

Title 

Credits 

Activity 


lb 


Enter Parameter Value 


Co u rs e s to r wh i oh d e p artm e nt? 



COMM 


JJ 


OK 


Cancel 


Field 

Table 

Sort 

Show 

Criteria 

or 


DeptCode 


CrsNum 


Title 




When Access asks for 
the value of the 
parameter, it uses the 
parameter’s name. 


Only records that satisfy 
the criteria are included 
in the results set. 


Courses 


Ascending 


si 1 pqryCourses : Select Query 




[Co u rs e s to r wh i ch d e p artm e nt?] 



Name the parameter [Courses 
for which department?]. 



Department 

J^urse number 

Title 

► 

IcommHa 

290 

Introduction to Quantative Decision \\< 


COMM 

291 

Applied Statistics in Business 


COMM 

351 

Financial Accounting 


COMM 

439 

Advanced Topics in Information Syst 

* 





I ^Homel M Previous | 5 of 11 | Next^ | 


10. Parameter Queries 


Tutorial exercises 


FIGURE 10.3: Using the value on an open form as a parameter in a query. 
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The current value in the DeptCode field on 
the form is used as a parameter in the query. 


I4I-h ome 1 Previous] 6 of 11 | Next ► ] 


































































































































































10. Parameter Queries 


Application to the assignment 


• Create a very simple form based on the 

Departments table and save it as frmDepart- 
ment s. 

• Leave the form open (in form view or design 
mode, it does not matter). 

• Open pqryCourses in design mode, place the 
cursor in the criteria row of the Deptcode field, 
and invoke the expression builder as shown in 
Figure 10.4. 

• Perform the steps shown in Figure 10.5 to create 
a parameter that references the Deptcode field 
On the f rmDepartments form. 

• Run the query. The results set should correspond 
to the department showing in the f rmDepart¬ 
ments form. 

• Move to a new record on the form. Notice that 
you have to requery the form ( Shift-F9 ) in order 
for the new parameter value to be used (see 
Figure 10.6). 



Although the naming syntax of objects in 
Access is tricky, it is not impossible to com¬ 
prehend. For example, the name 

Forms![frmDepartments]![DeptCode] 

consists of the following elements: Forms 
refers to a collection of Form objects; [f rm¬ 
Departments] is a specific instance of a 
Form object in the Forms collection; [Dept¬ 
code] is a Control belonging to the form. See 
Tutorial 14 for more information on the hierar¬ 
chy of objects used by Access. 


10.4 Application to the assignment 

You will use parameter queries as the basis for sev¬ 
eral action queries (see Tutorial 11) that process 
transactions against master tables. For now, simply 
create the parameter queries that take their criteria 
values from forms you have already created. 
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FIGURE 10.4: Invoke the builder to build a parameter. 
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10. Parameter Queries 


Application to the assignment 


FIGURE 10.5: Use the builder to select the name of the object you want to use as a parameter. 
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10. Parameter Queries Application to the assignment 

FIGURE 10.6: Requery the results set to reflect changes on the form. 


as 1 pqryCourses : Select Query 
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10. Parameter Queries 


Application to the assignment 


• Create a parameter query to show all the order 
details for a particular order. 

• Create a second parameter query to show all the 
shipment details for a particular shipment. 

Each order may result in a number of changes being 
made to the Backorders table. For some items in 
the order, more product is ordered than is actually 
shipped (i.e., a backorder is created). For other 
items, more product is shipped than is ordered (i.e., 
a backorder is filled). 

In Tutorial 15, you are supplied with a “shortcut” 
Visual Basic procedure that makes the changes to 
the Backorders table for you. However, the short¬ 
cut procedure requires a query that lists the changes 
that must be made to the Backorders table for a 
particular order. The requirements for this query are 
the following: 

• The name of the query is 

pqryItemsToBackOrder 


• It shows the change (positive or negative but not 
zero) in backorders for each item in a particular 
order. 

• The query consist of three fields: OrderiD, Pro¬ 
duct id and a calculated field Qty (i.e., the 
change in the back order for a particular product). 

• The name of the parameter is in this query is sim¬ 
ply [pOrderiD] . Since the value of this parame¬ 
ter will be set by the Visual Basic shortcut before 
the query is run, there is no need to set it to a 
value on a form. 



Since the query is accessed by a program, 
the name of the query and all the fields must 
be exactly as described above. In other 
words, you are given a precise specification 
for a database object that fills a role in a pro¬ 
cess designed and implemented by someone 
else. You will not understand how the query 
fits in until Tutorial 15. 
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Access Tutorial 11: Action Queries 


11.1 Introduction: Queries that 
change data 

11.1.1 What is an action query? 

All of the queries that you have created to this point 
have been variations of “select” queries. Select que¬ 
ries are used to display data but do not actually 
change the data in any way. 

Action queries, in contrast, are used to change the 
data in existing tables or make new tables based on 
the query's results set. The primary advantage of 
action queries is that they allow you to modify a large 
number of records without having to write Visual 
Basic programs. 

Access provides four different types of action que¬ 
ries: 

I . Make table — creates a new table based on the 
results set of the query; 

© Michael Brydon (brydon@unixg.ubc.ca) 
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II. Action Queries 

ues. There are at least four different ways of accom¬ 
plishing this task: 

1. Create a calculated field called NewCredit s that 
multiplies the value of Credits by 1.5 — The 
query containing the calculated field can be used 
in place of the courses table whenever credit 
information is required. Of course, the values 
stored in the courses table are still the old val¬ 
ues. Although there might be some advantages 
to keeping the old values, it may cause confusion 
about which values to use. In addition, the use of 
a calculated field creates a computational load 
that becomes larger as the number of courses 
increases. 

2. Go through the courses table record by record 
and manually change all the values — This 
approach is tedious and error prone. Further¬ 
more, it is simply impractical if the number of 
courses is large. 


2. Append — similar to a make-table query, except 
that the results set of the query is appended to an 
existing table; 

3. Update — allows the values of one or more fields 
in the result set to be modified; and, 

4. Delete — deletes all the records in the results set 
from the underlying table. 

Since the operation of all four types of action queries 
is similar, we will focus on update queries in this tuto¬ 
rial. 

11.1.2 Why use action queries? 

To motivate the examples in the first part of this tuto¬ 
rial, we are going to assume that the number of cred¬ 
its allocated to courses in certain departments need 
to be changed. For example, assume that you need 
to increase the number of credits for courses in the 
Commerce department by 1.5 times their current val- 
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Learning objectives 

3. Write a Visual Basic program to automate Step 2. 
This is a good approach; however, it clearly 
requires the ability to write Visual Basic pro¬ 
grams. 

4. Create an update query that (a) selects only 
those courses that require modification and (b) 
replaces the value of credits with credits * 
l . 5. — This approach is computationally efficient 
and allows you to work with the QBE editor rather 
than a programming language. 

11.2 Learning objectives 

□ What is an action query? Why would I want to 
use one? 

□ How do I make a backup copy of one of my 
tables? 

□ How to I undo (rollback) an action query once 
I have executed it? 
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11. Action Queries 


Tutorial exercises 


□ How do I update only certain records in a 
table? 

□ How do I create a button on a form? How do I 
make an action query execute when the 
button is pressed? 

11.3 Tutorial exercises 

11.3.1 Using a make-table query to create 
a backup 

Since action queries permanently modify the data in 
tables, it is a good idea to create a backup of the 
table in question before running the query. An easy 
way to do this is to use a make-table query. 

• Create a select query based on the courses 
table and save it as qryCoursesBackup. 

• Project the asterisk (*) into the query definition so 
that all the fields are included in the results set. 


• While still in query design mode, select Query > 
Make Table from the main menu and provide a 
name for the target table (e.g., coursesBackup) 
as shown in Figure 11.1. 

• Select Query > Run from the main menu to exe¬ 
cute the action query, as shown in Figure 11.2. 



Action queries do not execute until you explic¬ 
itly run them. Switching to datasheet mode 
only provides a preview of the results set. 


• Save the query. If you switch to the database win¬ 
dow, you will notice that the new make-table 
query has a different icon than the select queries. 


11.3.2 Using an update query to rollback 
changes 

Having a backup table is not much use without a 
means of using it to restore the data in your original 
table. In this section, you will use an update query to 
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11. Action Queries Tutorial exercises 

FIGURE 11.1: Use a make-table query to back up and existing table 
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11. Action Queries 


Tutorial exercises 


FIGURE 11.2: Run the make-table query. 
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You can switch to datasheet mode to view the results 
set. Note that this does not actually execute the query. 


Microsoft Access 


File E 



iew Insert 


o 


a * 


Query 


Run 


Tools Window Help 


ip qryCoursesBackup 


Courses 


DeptCode 

CrsNum 

Title 

Credits 

Activity 



Field 

Table 

Sort 

Show 

Criteria 


Show Table... 
Remove 

Select 

Crosstab 

Make Table... 

Update 

Append... 

Delete 

SQL Specific 

Parameters... 


Courses 


V 


To execute the query, you must select 
Query > Run. Alternatively, you can 
press the “run” (!) icon on the toolbar. 


Microsoft Access 
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The warning box reminds you that you 
are about to make permanent changes 
to the data in the database. 
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Tutorial exercises 


replace some of the values in your courses table 
with values from your coursesBackup table. 

• Create a new query based on the courses and 
CoursesBackup tables. 

• Since no relationship exists between these 
tables, create an ad hoc relationship within the 
query as shown in Figure 11.3. 

• Select Query > Update from the main menu. Note 
that this results in the addition of an Update To 
row in the query definition grid. 

• Project Credits into the query definition and fill 
in the Update To row as shown in Figure 11.4. 

• Save the query as qryRollbackCredits. 

Now is a good point to stop and interpret what you 
have done so far: 

1. By creating a relationship between the Courses 
table and its backup, you are joining together the 
records from both tables that satisfy the condi- 


FIGURE 11.3: Create an ad hoc relationship 
between the table and its backup. 


bs 1 Queryl : Select Query 


Courses 


T 

DeptCode 


CoursesBackup 


DeptCode 


CrsNum 

Title 

Credits 

Activity 





Drag the fields in the key on to their 
counterparts in the backup table. 

UCI IQ. 


© 


or: 


You cannot drag two fields at once or 
enforce referential integrity in a QBE 
relationship like you can in the main 
relationship editor. 
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11. Action Queries 


Tutorial exercises 


FIGURE 11.4: Fill in the Update To field. 


Select Query > Update to make 
the query an update query. 


is 1 Gueryl : Update Query 


| Courses 


CoursesBackup 

DeptCode 


Jc 

DeptCode 

CrsNum 


CrsNum 

Title 


Title 

Credits 


Credits 

Activity 


Activity 

lU 




Field: 
Table: 
Update To: 
Criteria: 



Credits 


Courses 


'Co u rs e s B acku p]. [ Cre d its 



Tb 


Usethe<table name>.<field name> 

syntax to disambiguate the field name. 


tion: 

Courses.DeptCode = 

CoursesBackup.DeptCode AND 
Courses.CrsNum = 

CoursesBackup.CrsNum. 

2. By projecting courses. Credits into the query, 
you are making it the target for the update. In 
other words, the values in courses. Credits 
are going to be modified by the update action. 

3. By setting the Update To field to Courses- 
Backup. credits, you are telling Access to 
replace the contents of courses .credits with 
the contents Of CoursesBackup . Credits. 

Whenever this query is run, it will replace whatever is 
in the Credits field of all the records in the 
Courses table with values from the backup. You will 
use this query to “rollback” updates made later on. 


| -^Homel H Previous! 7 of 16 | Next ► 1 


11. Action Queries 


Tutorial exercises 


11.3.3 Using an update query to make 
selective changes 

Now that you have an infrastructure for undoing any 
errors, you can continue with the task of updating 
credits for the Commerce department. 

• Create an update query based on the courses 
table and save it as qryUpdateCredits. 

• Set the Update To field to [Courses] *i. 5. 
Note that if you do not include the square brack¬ 
ets, Access will interpret Courses as a literal 
string rather than a field name. 



Since this particular query only contains one 
table, the <table name>.<field name> 

syntax is not required for specifying the 
Update To expression. 


• Since you only want to apply the change to Com¬ 
merce courses, enter a criterion for the Dept¬ 
Code field, as shown in Figure 11.5. 


FIGURE 11.5: Create an update query that 
updates a subset of the records. 


ii 1 qryUpdateCredits : Update Query 


Courses 


DeptCode 

CrsNum 

Title 

Credits 

Activity 


Set the Update 
To field to replace 
Credits with 
Credits X 1.5 


/ 


mum nnnniimmii mini fm 


Field: 
Table: 
Update To: 
Criteria: 
or: 




Credits ^ 

DeptCode 

Courses f 

Courses 

[Credits]*1.5 



"COMM" j 



. 



_ 


Add a criteria to limit the scope of 
the update. Note that DeptCode is 
not changed in any way by this query. 
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11. Action Queries 


Tutorial exercises 


• Run the query and verify that update has been 
performed successfully. 


11.3.5 Attaching action queries to 
buttons 


11.3.4 Rolling back the changes 

While testing the qryUpdateCredits query, your 
exuberance may have led you to execute it more 
than once. To return the Courses table to its state 
before any updates, all you need to do it run your 
rollback query. 

• Run qryRoiiback credits by double-clicking its 
icon in the database window. 



Once an action query is created, it has more 
in common with subroutines written in Visual 
Basic than standard select queries. As such, it 
is best to think of action queries in terms of 
procedures to be executed rather than virtual 
tables or views. Double-clicking an action 
query executes it. 


As a designer, you should not expect your users to 
understand your query naming convention, rum¬ 
mage through the queries listed in the database win¬ 
dow, and execute the queries that need to be 
executed. As such, it is often useful to create buttons 
on forms and “attach” the action queries to the but¬ 
tons. When the button is pressed, the query is exe¬ 
cuted. 

Although we have not yet discussed buttons (or 
events in general), the button wizard makes the cre¬ 
ation of this type of form object straightforward. 

• Modify qryUpdateCredits so that it updates 
only those departments matching the Deptcode 
value in the f rmDepartments table (see 
Figure 11.6). 

• Save the resulting action parameter query as 

pqryUpdateCredits and close it. 
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11. Action Queries Tutorial exercises 

FIGURE 11.6: Create an action parameter query to update credits for a particular department. 



The update operation 

The criterion limits the scope of the 

specifies the action to 

update to those records matching 

perform on the records. 

the current parameter value 
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11. Action Queries 


Application to the assignment 


• Switch to the design view Of frmDepartments 
and add a button as shown in Figure 11.7. 

• Attach the pqryUpdateCredits query to the 
button as shown in Figure 11.8. 

• Provide a caption and a name for the button as 
shown in Figure 11.9. 

• Switch to form view. Press the button to run the 
query (alternatively, use the shortcut key by 
pressing Alt-U) as shown in Figure 11.10. 

11.4 Application to the assignment 

11.4.1 Rolling back your master tables 

As you begin to implement the transaction process¬ 
ing component of your system, it is worthwhile to 
have a means of returning your master tables to their 
original state (i.e., their state when you started devel¬ 
oping the system). 


• Create backup copies of your Products and 
Backorders tables using make-tables queries. 
Save these queries but note that they only need 
to be run once. 

• Create a rollback query that allows you to return 
your Products table to its original state. 

Rolling back the Backorders table is more complex 
than rolling back the Products table. This is 
because we are making the assumption that no 
products are ever added or deleted to the system. As 
such, all the information needed for the rollback is in 
the backup copy Of Products. 

In contrast, records are added to the Backorders 
table on a regular basis. As a result, the Back¬ 
orders table and its backup may contain a different 
number of records. If so, the match-and-replace pro¬ 
cess used for rolling back Products is inappropri¬ 
ate. 
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11. Action Queries Application to the assignment 

FIGURE 11.7: Add a button to the form using the button wizard. 


If there is insufficient space for a button, drag 
the border of the detail section to the right 


M frmDepartments : Form 


i 1 i 1 i 


+ Form Header 


* Detail 


D e p £ irtm e nt c □ d e D e ptCc 
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Building 


Buildim 
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^ Form Footer 
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i 1 10' i 1 11 
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■ Command 6 J 
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A 
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H 

m 

Is 

_ j. 

i 

o 

IS 

m 

+ D 

“l —1 

|is| 


1 1 


□ x 


13' i 



Select the “command button” tool and click 
on an appropriate location on the form detail 
section. The button wizard should appear 


lb 


Ensure that the 
wizard button in the 
toolbox is 
depressed (wizards 
are activated). 
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11. Action Queries 


Application to the assignment 


FIGURE 11.8: Use the wizard to attach an action query to the button. 


Command Button Wizard 



What action do you want to happen when the button is pressed? 
Different actions are available for each category. 


Categories: 


Actions: 


Record Navigation 


Record Operations 
Form Operations 
Report Operations 
Application 



Miscellaneous 


Command Button Wizard 


lb 


The wizard lists all the available 
queries (including non-action queries). 
Select pqryUpdateCredit s. 


Buttons can be created to 
perform many different actions 
in Access. The button wizard 
organizes these actions into 
categories. Select 
Miscellaneous > Run Query. 



What query would you like the command button to run? 


pqryCourses 


Up date Credits 


qryCatalogNum 
qry Course Lengths 
qryCourses 
q ry Co u rs e s B acku p 
qryRollbackCredits 
qry Up date Credits 
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11. Action Queries 


Application to the assignment 


FIGURE 11.9: Use the wizard to attach a query to a button (continued) 


You can show either a picture (icon) or a caption 
on the button. Enter a suitable caption. 


© 


Command Button Wizard 



Do you want text or a picture on the buttojj 



If ydunhoose T ext, you canjype the text to display. 
Picture^VOMcan clbaJfcBfowse to find a picture to disp 


choose 


** T ext: 


& Update Credits 


c Picture: I Append Query 



Browse... 


Including an ampersand (&) in 
the caption creates a shortcut 
key from the letter immediately 
following the ampersand. 
Shortcut keys can be invoked 
using the Alt-<letter> 
combination (the letter is 
underlined). In this case, Alt-U 
moves the focus directly to the 
button. 


Command Button Wizard 



What do you want to name the button? 

A meaningful name will help you to refer to the button later. 
| cm d U p d ate Cred its 


That's all the inform 
command button. 




the wizard needs to create your 


^9 Provide a meaningful name for the 
button. The cmd prefix indicates a 
command button. 
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11. Action Queries 


Application to the assignment 


FIGURE 11.10: Execute the action query by pressing the button. 


Press the button to execute the action query 
(or press Alt-U to use the shortcut). 


B1 Departments 


D e p artm e nt co d e CO M M 






D e p artm e nt n am e Co rn m e roe an dBusinessAdmini stn 


Update Credits 


Building ANGU 


Microsoft Access 


m 


A You are about to run an update query that will modify data in your table. 

Are you sure you want to run this action query? 

Cl i ok H e I p f o r i nf o rm ati o n o n h ow to p re ve nt th i s m e s s ag e fro rn displ ay i n g eve ry time 
you run an action query. 


Record: Hhll 

«► ii h wmmmam 




Yes 

No 

Help 1 
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11. Action Queries 


Application to the assignment 


The easiest way to rollback the Backorders table is 
to delete all the records it contains and use an 
append query to replace the records from the 
backup. 

• Open your Backorders table in datasheet mode 
and select Edit > Select All Records from the 
menu (alternatively, press Control-A) 

• Press the Delete key. 

• Create an append query that adds the records 
in the backup table to the Backorders table. 

Once you learn the Access macro language or 
Visual Basic for Applications, you will be able to write 
a small procedure to execute these steps for you. 

For the assignment, however, this “manual rollback” 
is sufficient. 

11.4.2 Processing transactions 

You are now in a position to combine parameter que¬ 
ries and action queries into parameter-action que¬ 


ries. These queries will allow you to perform 
reasonably complex transaction processing opera¬ 
tions on your master tables. 

• Create an update query to add all products in a 
shipment to inventory. 


A Note that this query should only process ship- 

IA ment details for the current shipment (i.e., it 
should be based on a parameter query similar 
to the one you created in Section 10.4). 

• Create a button on the shipments form to perform 
this update. 

• Create an update query to subtract items from 
inventory when you process an order from your 
customers. Do not attach this query to a button at 
this point. 



This query should only process order details 
from the current order. 
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Access Tutorial 12: An Introduction to Visual Basic 


12.1 Introduction: Learning the 
basics of programming 

Programming can be an enormously complex and 
difficult activity. Or it can be quite straightforward. In 
either case, the basic programming concepts remain 
the same. This tutorial is an introduction to a handful 
of programming constructs that apply to any “third 
generation” language, not only Visual Basic for 
Applications (VBA). 

A Strictly speaking, the language that is 

. included with Access is not Visual Basic—it is 
a subset of the full, stand-alone Visual Basic 
language (which Microsoft sells separately). 

In Access version 2.0, the subset is called 
“Access Basic”. In version 7.0, it is slightly 
enlarged subset called “Visual Basic for Appli¬ 
cations” (VBA). However, in the context of the 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 25-Aug-1997 

12. An Introduction to Visual Basic 

In the second part of the tutorial, you are going to 
create a couple of VBA modules to explore looping, 
conditional branching, and parameter passing. 

12.2 Learning objectives 

□ What is the debug/immediate window? How 
do I invoke it? 

□ What are statements, variables, the 
assignment operator, and predefined 
functions? 

□ How do I create a module containing VBA 
code? 

□ What are looping and conditional branching? 
What language constructs can I use to 
implement them? 

□ How do I use the debugger in Access? 

□ What is the difference between an interpreted 
and compiled programming language? 


simple programs we are writing here, these 
terms are interchangeable. 

12.1.1 Interacting with the interpreter 

Access provides two ways of interacting with the 
VBA language. The most useful of these is through 
saved modules that contain VBA procedures. These 
procedures (subroutines and functions) can be run to 
do interesting things like process transactions 
against master tables, provide sophisticated error 
checking, and so on. 

The second way to interact with VBA is directly 
through the interpreter. Interpreted languages are 
easier to experiment with since you can invoke the 
interpreter at any time, type in a command, and 
watch it execute. In the first part of this tutorial, you 
are going to invoke Access’ VBA interpreter and exe¬ 
cute some very simple statements. 
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Learning objectives 

12.3 Tutorial exercises 

12.3.1 Invoking the interpreter 

• Click on the module tab in the database window 
and press New. 

This opens the module window which we will use in 
Section 12.3.3. You have to have a module window 
open in order for the debug window to be available 
from the menu. 

• Select View > Debug Window horn the main 
menu. Note that Control-G can be used in ver¬ 
sion 7.0 and above as a shortcut to bring up the 
debug window. 

A in version 2.0, the “debug” window is called 
. the “immediate” window. As such, you have to 
use View > Immediate Window. The term 
debug window will be used throughout this 
tutorial. 
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12. An Introduction to Visual Basic 


Tutorial exercises 


12.3.2 Basic programming constructs 

In this section, we are going to use the debug win¬ 
dow to explore some basic programming constructs. 

12.3.2.1 Statements 

Statements are special keywords in a programming 
language that do something when executed. For 
example, the Print statement in VBA prints an 
expression on the screen. 

• In the debug window, type the following: 

Print "Hello world!"J 

(the J symbol at the end of a line means “press the 
Return or Enter key”). 


12.3.2.2 Variables and assignment 

A variable is space in memory to which you assign a 
name. When you use the variable name in expres¬ 
sions, the programming language replaces the vari¬ 
able name with the contents of the space in memory 
at that particular instant. 

• Type the following: 

s = "Hello"J 
? s & " world"J 
? "s" & " world"J 

In the first statement, a variable s is created and the 
string Hello is assigned to it. Recall the function of 
the concatenation operator (&) from Section 4.4.2. 



In VBA (as in all dialects of BASIC), the ques¬ 
tion mark (?) is typically used as shorthand for 
the Print statement. As such, the statement: 
? "Hello world! "J is identical to the 
statement above. 



Contrary to the practice in languages like C 
and Pascal, the equals sign (=) is used to 
assign values to variables. It is also used as 
the equivalence operator (e.g., does x = y?). 
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12. An Introduction to Visual Basic 


Tutorial exercises 


When the second statement is executed, VBA recog¬ 
nizes that s is a variable, not a string (since it is not 
in quotations marks). The interpreter replaces s with 
its value (Hello) before executing the Print com¬ 
mand. In the final statement, s is in quotation marks 
so it is interpreted as a literal string. 

A Within the debug window, any string of char- 
. acters in quotations marks (e.g., "comm") is 
interpreted as a literal string. Any string with¬ 
out quotation marks (e.g., comm) is interpreted 
as a variable (or a field name, if appropriate). 
Note, however, that this convention is not uni¬ 
versally true within different parts of Access. 


is a function that is provided as part of the program¬ 
ming environment. 

For example, cos (x) is a predefined function in 
many computer languages—it takes some number x 
as an argument, does some processing to find its 
cosine, and returns the answer. Note that since this 
function is predefined, you do not have to know any¬ 
thing about the algorithm used to find the cosine, you 
just have to know the following: 

1. what to supply as inputs (e.g., a valid numeric 
expression representing an angle in radians), 

2. what to expect as output (e.g., a real number 
between -1.0 and 1.0). 


12.3.2.3 Predefined functions 

In computer programming, a function is a small pro¬ 
gram that takes one or more arguments (or param¬ 
eters) as input, does some processing, and returns 
a value as output. A predefined (or built-in) function 



The on-line help system provides these two 
pieces of information (plus a usage example 
and some additional remarks) for all VBA pre¬ 
defined functions. 
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12. An Introduction to Visual Basic 


Tutorial exercises 


In this section, we are going to explore some basic 
predefined functions for working with numbers and 
text. The results of these exercises are shown in 
Figure 12.1 . 

• Print the cosine of 2tz radians: 

pi = 3.14159J 
? cos(2*pi)J 

• Convert a string of characters to uppercase: 

s = "basic or cobol"J 
? UCase(s)J 

• Extract the middle six characters from a string 
starting at the fifth character: 

? mid (s,5,6)J 

12.3.2.4 Remark statements 

When creating large programs, it is considered good 
programming practice to include adequate internal 
documentation—that is, to include comments to 
explain what the program is doing. 


FIGURE 12.1: Interacting with the Visual Basic 

interpreter. 



pi = 3.14159 
? cos(2*pi) 
0.999999999985917 


The argument contains 
an expression. 


s = "basic or cobol 
? UCase(s) 

BASIC OR COBOL 

? mid (s,5,6) 
c or c 



UCase () converts a 
string to uppercase. 

Mid () extracts 
characters from the 
string defined earlier. 
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12. An Introduction to Visual Basic 


Tutorial exercises 


Comment lines are ignored by the interpreter when 
the program is run. To designate a comment in VBA, 
use an apostrophe to start the comment, e.g.: 

' This is a comment line! 

Print "Hello" 'the comment starts 
here 

The original REM (remark) statement from BASIC 
can also be used, but is less common. 

REM This is also a comment (remark) 


FIGURE 12.2: The declarations page of a Visual 

Basic module. 


v Module 

1 : Module 


mzi 

Object: 

(General) 

Proc: (declarations) 


Option Compare Database 
Option Explicit 


12.3.3 Creating a module 

• Close the debug window so that the declaration 
page of the new module created in 
Section 12.3.3 is visible (see Figure 12.2). 

The two lines: 

Option Compare Database 
Option Explicit 

are included in the module by default. The Option 
Compare statement specifies the way in which 


strings are compared (e.g., does uppercase/ lower¬ 
case matter?). The Option Explicit statement 
forces you to declare all your variables before using 
them. 

A in version 2.0, Access does not add the 

. Option Explicit statement by default. As 
such you should add it yourself. 
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12. An Introduction to Visual Basic 


Tutorial exercises 


A module contains a declaration page and one or 
more pages containing subroutines or user-defined 
functions. The primary difference between subrou¬ 
tines and functions is that subroutines simply exe¬ 
cute whereas functions are expected to return a 
value (e.g., cos ()). Since only one subroutine or 
function shows in the window at a time, you must 
use the Page Up and Page Down keys to navigate 
the module. 

A The VBA editor in version 8.0 has a number of 
. enhancements over earlier version, including 
the capability of showing multiple functions 
and subroutines on the same page. 


12.3.4 Creating subroutines with looping 
and branching 

In this section, you will explore two of the most pow¬ 
erful constructs in computer programming: looping 
and conditional branching. 

• Create a new subroutine by typing the following 
anywhere on the declarations page: 

Sub LoopingTest()J 

Notice that Access creates a new page in the mod¬ 
ule for the subroutine, as shown in Figure 12.3. 

12.3.4.1 Declaring variables 

When you declare a variable, you tell the program¬ 
ming environment to reserve some space in memory 
for the variable. Since the amount of space that is 
required is completely dependent on the type of data 
the variable is going to contain (e.g., string, integer, 
Boolean, double-precision floating-point, etc.), you 
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12. An Introduction to Visual Basic 


Tutorial exercises 


FIGURE 12.3: Create a new subroutine. 


v Module! : Module 

ZEE 

Object: (General) 

Proc: 

LoopingTest 

_ 


Sub LoopingTest() 


End Sub 


© 


You can use the procedure 
combo box to switch between 
procedures in a module. 


have to include data type information in the declara¬ 
tion statement. 

In VBA, you use the Dim statement to declare vari¬ 
ables. 

• Type the following into the space between the 

Sub . . . End Sub pair: 

Dim i as integer 
Dim s as string 


• Save the module as basTesting. 

One of the most useful looping constructs is For 
<condition>.. . Next. All statements between 
the For and Next parts are repeated as long as the 
<condition> part is true. The index i is automati¬ 
cally incremented after each iteration. 

• Enter the remainder of the LoopingTest pro¬ 
gram: 

s = "Loop number: " 

For i = 1 To 10 

Debug.Print s & i 
Next i 

• Save the module. 



It is customary in most programming lan¬ 
guages to use the Tab key to indent the ele¬ 
ments within a loop slightly. This makes the 
program more readable. 
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12. An Introduction to Visual Basic 


Tutorial exercises 


Note that the Print statement within the subroutine 
is prefaced by Debug. This is due to the object-ori¬ 
ented nature of VBA which will be explored in greater 
detail in Tutorial 14. 

12.3.4.2 Running the subroutine 

Now that you have created a subroutine, you need to 
run it to see that it works. To invoke a subroutine, you 
simply use its name like you would any statement. 

• Select View > Debug Window from the menu (or 
press Control-G in version 7.0). 

• Type: LoopingTest J in the debug window, as 
shown in Figure 12.4. 

12.3.4.3 Conditional branching 

We can use a different looping construct, Do until 
<condition>. .. Loop, and the conditional 
branching construct, if <condition> Then... 
Else, to achieve the same result. 


FIGURE 12.4: Run the LoopingTest 
subroutine in the debug window. 


v basTesting : Module 


Object: j (General) 


Proc: 


LoopingTest 


Sub LoopingTestf) 

Dim i As Integer 
Dim s As String 

s = "Loop number: 

For i = 1 To 10 

Debug.Print s/& i 
Next i 

End Sub 


Debug Window 



ngTest 

number 

number 

number 

number 

number 

number 

number 

number 

number 

number 


1 

2 

3 

4 

5 

6 

7 

8 

9 

10 


Invoke the LoopingTest subroutine 
by typing its name in the debug window. 
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12. An Introduction to Visual Basic 

• Type the following anywhere under the End Sub 
statement in order to create a new page in the 
module: 

Sub BranchingTestJ 

• Enter the following program: 

Dim i As Integer 
Dim s As String 
Dim intDone As Integer 
s = "Loop number: " 
i = 1 

intDone = False 
Do Until intDone = True 
If i > 10 Then 

Debug.Print "All done" 
intDone = True 
Else 

Debug.Print s & i 
i = i + 1 
End I f 


Tutorial exercises 

Loop 

• Run the program 

12.3.5 Using the debugger 

Access provides a rudimentary debugger to help you 
step through your programs and understand how 
they are executing. The two basic elements of the 
debugger used here are breakpoints and stepping 
(line-by-line execution). 

• Move to the s = "Loop number: " line in your 
BranchingTest subroutine and select Run > 
Toggle Breakpoint from the menu (you can also 
press F9 to toggle the breakpoint on a particular 
line of code). 

Note that the line becomes highlighted, indicating the 
presence of an active breakpoint. When the program 
runs, it will suspend execution at this breakpoint and 
pass control of the program back to you. 
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12. An Introduction to Visual Basic 


Tutorial exercises 


• Run the subroutine from the debug window, as 
shown in Figure 12.5. 

• Step through a couple of lines in the program 
line-by-line by pressing F8. 

By stepping through a program line by line, you can 
usually find any program bugs. In addition, you can 
use the debug window to examine the value of vari¬ 
ables while the program’s execution is suspended. 

• click on the debug window and type 
? iJ 

to see the current value of the variable i. 

12.3.6 Passing parameters 

In the BranchingTest subroutine, the loop starts 
at 1 and repeats until the counter i reaches 10. It 
may be preferable, however, to set the start and fin¬ 
ish quantities when the subroutine is called from the 
debug window. To achieve this, we have to pass 
parameters (or arguments) to the subroutine. 


FIGURE 12.5: Execution of the subroutine is 
suspended at the breakpoint. 


v : basTesting : Module 




Object: (General) 


Proc: 


BranchingTest 


Sub BranchingTest() 

Dim i As Integer 

Dim s As String 

Dim intDone As Integer 




Debug Window 


| [u ri ivO_v?. m d b]. b asT e sti 


s = "Loop number: 


i = 1 
intDone = False 


\ 


BranchingTest 


Do Until intDc (?) 

If i > ID 
Debug . 
intDor 

Else 

Debug.Print s & i 


The outlined box indicates the 
current location of the 
interpreter in the program. Press 
F8 to execute the line of code. 
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Tutorial exercises 


The main difference between passed parameters 
and other variables in a procedure is that passed 
parameters are declared in the first line of the sub¬ 
routine definition. For example, following subroutine 
declaration 

Sub BranchingTest (intStart as 
Integer, intStop as Integer) 

not only declares the variables intstart and 
intstop as integers, it also tells the subroutine to 
expect these two numbers to be passed as parame¬ 
ters. 

To see how this works, create a new subroutine 
Called ParameterTest based on Branch- 
ingTest. 

• Type the declaration statement above to create 
the ParameterTest subroutine. 

• Switch back to BranchingTest and highlight all 
the code except the sub and End sub state¬ 
ments, as shown in Figure 12.6. 


FIGURE 12.6: Highlight the code to copy it. 


| I basTesting : Module 

Object: 

i - 

(General) 


Sub BranchingTest() 


Dim i As Integer 
Dim s As String 

1 

Dim intDone As I 

nteger 

l 

s = "Loop number 

• " m 

1 


intDone = False 





Do Until intDone = True 
If i > 10 Then ■■ 


Loo 


Debug.Print "Al l done" 
i ntDone = True 

Else 

Debug .Prin t s & i 
i = i + 

End If 


End Sub 
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12. An Introduction to Visual Basic 


Tutorial exercises 


• Copy the highlighted code to the clipboard (Con¬ 
trol-Insert), switch to ParameterTest, and 
paste the code ( Shift-Insert ) into the Parame¬ 
terTest procedure. 

To incorporate the parameters into ParameterT¬ 
est, you will have to make the following modifica¬ 
tions to the pasted code: 

• Replace i = 1 with i = intStart. 

• Replace i > lOwithi > intstop. 

• Call the subroutine from the debug window by 
typing: 

ParameterTest 4, 12J 



If you prefer enclosing parameters in brack¬ 
ets, you have to use the call <sub 

name>(parameter^ . parameter n ) 

syntax. For example: 

Call ParameterTest(4,12)J 


12.3.7 Creating the Min () function 

In this section, you are going to create a user- 
defined function that returns the minimum of two 
numbers. Although most languages supply such a 
function, Access does not (the Min () and Max () 
function in Access are for use within SQL statements 
only). 

• Create a new module called basutilities. 

• Type the following to create a new function: 

Function MinValue(nl as Single, n2 
as Single) as SingleJ 

This defines a function called MinValue that returns 
a single-precision number. The function requires two 
single-precision numbers as parameters. 



Since a function returns a value, the data type 
of the return value should be specified in the 
function declaration. As such, the basic syn¬ 
tax of a function declaration is: 
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Function <function 
name>(parameter^ As <data type>, 
...f parameter n As <data type>) As 
<data type> 

The function returns a variable named 

<function name>. 

• Type the following as the body of the function: 

If nl <= n2 Then 
MinValue = nl 
Else 

MinValue = n2 
End I f 

• Test the function, as shown in Figure 12.7. 


Discussion 

12.4 Discussion 

12.4.1 Interpreted and compiled 
languages 

VBA is an interpreted language. In interpreted lan¬ 
guages, each line of the program is interpreted (con¬ 
verted into machine language) and executed when 
the program is run. Other languages (such as C, 
Pascal, FORTRAN, etc.) are compiled, meaning 
that the original (source) program is translated and 
saved into a file of machine language commands. 
This executable file is run instead of the source 
code. 

Predictably, compiled languages run much faster 
then interpreted languages (e.g., compiled C++ is 
generally ten times faster than interpreted Java). 
However, interpreted languages are typically easier 
to learn and experiment with. 


I ome 1 Previous] 14 of 16 | Next ► ] 
















12. An Introduction to Visual Basic 


Discussion 


FIGURE 12.7: Testing the MinVaiue () function. 


basUtilities : Module 


Object: j (General) 


Implement the MinVaiue () 
function using conditional branching. 


J 


Proc: 


MinVaiue 


Funct 


ion Mir>balue(n1 As Single, n2 As Single) As Single Test the function by passing it 

ft various parameter values. 


If nl <= n2 Then 
MinUalue = nl 
Else 

MinUalue = n2 
End If 


End 


F Me t i 


© 


ion 


S Debug Window 


|<Ready> 


? MinUalue(8,12) 

8 

? MinUalue(0.001, -0.001) 
- 0.001 




© According to the function 
declaration, MinVaiue () 
expects two single-precision 
numbers as parameters. 

Anything else generates an error. 


? MinUaluef"ten", "twelve") 


Microsoft Access 




& 


Run-time error '13': 


Type mismatch 


These five lines could be replaced with one line: 

MinVaiue = iif (nl <= n2, nl, n2) 


[ OK 


Help 
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12.5 Application to the assignment 

You will need a MinVaiue () function later in the 
assignment when you have to determine the quantity 
to ship. 

• Create a basUtilities module in your assign¬ 
ment database and implement a MinVaiue () 
function. 



To ensure that no confusion arises between 
your user-defined function and the built-in 
SQL Min () function, do not call you function 


Min ( ) . 


Application to the assignment 
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Access Tutorial 13: Event-Driven Programming 

Using Macros 


13.1 Introduction: What is event- 
driven programming? 

In conventional programming, the sequence of oper¬ 
ations for an application is determined by a central 
controlling program (e.g., a main procedure). In 
event-driven programming, the sequence of opera¬ 
tions for an application is determined by the user’s 
interaction with the application’s interface (forms, 
menus, buttons, etc.). 

For example, rather than having a main procedure 
that executes an order entry module followed by a 
data verification module followed by an inventory 
update module, an event-driven application remains 
in the background until certain events happen: when 
a value in a field is modified, a small data verification 
program is executed; when the user indicates that 


the order entry is complete, the inventory update 
module is executed, and so on. 

Event-driven programming, graphical user interfaces 
(GUIs), and object-orientation are all related since 
forms (like those created in Tutorial 6) and the 
graphical interface objects on the forms serve as the 
skeleton for the entire application. To create an 
event-driven application, the programmer creates 
small programs and attaches them to events associ¬ 
ated with objects, as shown in Figure 13.1 . In this 
way, the behavior of the application is determined by 
the interaction of a number of small manageable pro¬ 
grams rather than one large program. 


© Michael Brydon (brydon@unixg.ubc.ca) 
Last update: 25-Aug-1997 
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13. Event-Driven Programming Using Macros 


Introduction: What is event-driven programming? 


FIGURE 13.1: In a trigger, a procedure is 
attached to an event. 


interface object A 

cmdUpdateCredit s 

properties 

Caption 

Enabled 


An object, such as the 
button created in 
Section 11.3.5, has 

predefined properties and 
events. For a button, the 
most important event is 

On Click. 


events 


On 

On 


Click 
Got F 


ocus 


A procedure (such as an 
action query, macro, or VBA 
function or subroutine) can be 
attached to an event. When 
the event occurs, the 
procedure is executed. 


procedure 



13.1.1 Triggers 

Since events on forms “trigger” actions, event/proce¬ 
dure combinations are sometimes called triggers. 

For example, the action query you attached to a but¬ 
ton in Section 11.3.5 is an example of a simple, one- 
action trigger. However, since an action query can 
only perform one type of action, and since you typi¬ 
cally have a number of actions that need to be per¬ 
formed, macros or Visual Basic procedures are 
typically used to implement a triggers in Access. 

13.1.2 The Access macro language 

As you discovered in Tutorial 12, writing simple VBA 
programs is not difficult, but it is tedious and error- 
prone. Furthermore, as you will see in Tutorial 14, 
VBA programming becomes much more difficult 
when you have to refer to objects using the naming 
conventions of the database object hierarchy. As a 
consequence, even experienced Access program- 
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13. Event-Driven Programming Using Macros 

mers often turn to the Access macro language to 
implement basic triggers. 

The macro language itself consists of 40 or so com¬ 
mands. Although it is essentially a procedural lan¬ 
guage (like VBA), the commands are relatively high 
level and easy to understand. In addition, the macro 
editor simplifies the specification of the action argu¬ 
ments (parameters). 

13.1.3 The trigger design cycle 

To create a trigger, you need to answer two ques¬ 
tions: 

1. What has to happen? 

2. When should it happen? 

Once you have answered the first question (“what”), 
you can create a macro (or VBA procedure) to exe¬ 
cute the necessary steps. Once you know the 
answer to the second question (“when”), you can 


Learning objectives 

attach the procedure to the correct event of the cor¬ 
rect object. 

A Selecting the correct object and the correct 
. event for a trigger is often the most difficult 
part of creating an event-driven application. It 
is best to think about this carefully before you 
get too caught up in implementing the proce¬ 
dure. 

13.2 Learning objectives 

□ What is event-driven programming? What is a 
trigger? 

□ How do I design a trigger? 

□ How does the macro editor in Access work? 

□ How do I attach a macro to an event? 

□ What is the SetValue action? How is it used? 
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□ How do I make the execution of particular 
macro actions conditional? 

□ What is a switchboard and how do I create 
one for my application? 

□ How to I make things happen when the 
application is opened? 

□ What are the advantages and disadvantages 
of event-driven programming? 

13.3 Tutorial exercises 

In this tutorial, you will build a number of very simple 
triggers using Access macros. These triggers, by 
themselves, are not particularly useful and are 
intended for illustrative purposes only. 


Tutorial exercises 

13.3.1 The basics of the macro editor 

In this section, you are going to eliminate the warn¬ 
ing messages that precede the trigger you created 
Section 11.3.5. 

As such, the answer to the “what” question is the fol¬ 
lowing: 

1. Turn off the warnings so the dialog boxes do not 
pop up when the action query is executed; 

2. Run the action query; and, 

3. Turn the warnings back on (it is generally good 
programming practice to return the environment 
to its original state). 

Since a number of things have to happen, you can¬ 
not rely on an action query by itself. You can, how¬ 
ever, execute a macro that executes several actions 
including one or more action queries. 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


• Select the Macros tab from the database window 
and press New. This brings up the macro editor 
shown in Figure 13.2. 

• Add the three commands as shown in 

Figure 13.3. Note that the openQuery command 
is used to run the action query. 

• Save the macro as mcrupdatecredits and 
close it. 

13.3.2 Attaching the macro to the event 

The answer to the “when” question is: When the 
cmdUpdateCredit s button is pressed. Since you 
already created the button in Section 11.3.5, all you 
need to do is modify its On Click property to point the 
mcrUpdateCredits macro. 

• Open f rmDepartments in design mode. 

• Bring up the property sheet for the button and 
scroll down until you find the On Click property, 
as shown in Figure 13.4. 


FIGURE 13.4: Bring up the On Click property for 

the button. 



a 


Update Credits 
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Tutorial exercises 


FIGURE 13.2: The macro editor. 


Macro actions can be selected from a list. The 
SetWarnings command is used to turn the warning 
messages (e.g., before you run an action query) on and off. 


In the comment column, you can 
document your macros as required 


Multiple commands are 
executed from top to 
bottom. 


Most actions have one or 
more arguments that 
determine the specific 
behavior of the action. In 
this case, the 
SetWarnings action is 
set to turn warnings off. 



*?. Macrol : Macro 


Action 


SetWarnings 


Comment 


The area on the right 
displays information about 
the action. 



Action Arguments 


Warnings On 


No 


dJ 


Turns all system messages on or 
off. Prevents modal warnings from 
stopping the macro (although 
error messages and dialogs that 
require user input still appear). 
This has the same effect as 
pressing Enter in each message 
box (typically an OK or Yes). 
Press FI for help on this action. 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


FIGURE 13.3: Create a macro that answers the “what” question. 


Add the three commands to 
the macro. 



The arguments for the two 
SetWarnings actions 
are straightforward. For the 
OpenQuery command, 
you can select the query to 
open (or run) from a list. 
Since this is an action 
query, the second and third 
arguments are not 
applicable. 


mcrUp date Credits : Macro 


Action 




SetWarnings 


OpenQuery 


SetWarnings 


Query Name 
View 

Data Mode 


Comment 




Action Arguments 




pqryUpdate Credits 

w 

Datasheet 


Edit 



Selectthe name of the query to 
open. The list shows all queries in 
the current database. Required 
argument. Press FI for help on 
this argument. 
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Tutorial exercises 


• Press the builder button ( - ) beside the existing 
procedure and look at the VBA subroutine cre¬ 
ated by the button wizard. Most of this code is for 
error handling. 



Unlike the stand-along VBA modules you cre¬ 
ated in Tutorial 12, this module (collection of 
functions and subroutines) is embedded in 

the f rmDepartments form. 


• Since you are going to replace this code with a 
macro, you do not want it taking up space in your 
database file. Highlight the text in the subroutine 
and delete it. When you close the module win¬ 
dow, you will see the reference to the “event pro¬ 
cedure” is gone. 

• Bring up the list of choice for the On Click prop¬ 
erty as shown in Figure 13.5. Select mcrUp- 

dateCredits. 


FIGURE 13.5: Select the macro to attach to the 

On Click property. 



if Command Button: cmdUpdateCredits 


Format | Data | Event | Other 


All 


Shortcut Menu Bar. 

ControlTip Text.. 

Help Context Id. 0 

q Press the arrow to get a list 

o 


of available macros 


On Got Focus 
On Lost Focus 
On Click. 


On Dbl Click. [Event Procedure] 

On Mouse Down ... 


On Mouse Move 
On Mouse Up .. 
On Key Down ... 
On Key Up .. 



mcrUp date Credits 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


• Switch to form view and press the button. Since 
no warnings appear, you may want to press the 
button a few times (you can always use your roll¬ 
back query to reset the credits to their original 
values). 

13.3.3 Creating a check box to display 
update status information 

Since the warning boxes have been disabled for the 
update credits trigger, it may be useful to keep track 
of whether courses in a particular department have 
already been updated. 

To do this, you can add a field to the Departments 
table to store this “update status” information. 

• Edit the Departments table and add a Yes/No 
field called CrUpdated. 

A If you have an open query or form based on 
. the Departments table, you will not be able 


to modify the structure of the table until the 
query or form is closed. 

• Set the Caption property to Credits updated? 
and the Default property to No as shown in 
Figure 13.6. 

Changes made to a table do not automatically carry 
over to forms already based on that table. As such, 
you must manually add the new field to the depart¬ 
ments form. 

• Open f rmDepartments in design mode. 

• Make sure the toolbox and field list are visible. 
Notice that the new field (CrUpdated) shows up 
in the field list. 

• Use the same technique for creating combo 
boxes to create a bound check box control for the 
yes/no field. This is shown in Figure 13.7. 
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FIGURE 13.6: Add a field to the Departments 
table to record the status of updates. 



13.3.4 The Setvaiue command 

So far, you have used two commands in the Access 
macro language: SetWarnings and OpenQuery. In 


Tutorial exercises 

this section, you are going to use one of the most 
useful commands— setvaiue —to automatically 
change the value of the CrUpdated check box. 

• Open your mcrUpdateCredits macro in design 
mode and add a setvaiue command to change 
the CrUpdated check box to Yes (or True, if 
you prefer). This is shown in Figure 13.8. 

• Save the macro and press the button on the form. 
Notice that the value of the check box changes, 
reminding you not to update the courses for a 
particular department more than once. 

13.3.5 Creating conditional macros 

Rather than relying on the user not to run the update 
when the check box is checked, you may use a con¬ 
ditional macro to prevent an update when the 
check box is checked. 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


FIGURE 13.7: Add a check box control to keep track of the update status. 


S3 frmDepartments : Form 


l l'1 l l l 2' |, 3' |l 4 l| '5 l| '6' |l 7' |l S ,|l 9 I| '10' 1 


Form Header 


* Detail 


D e p i: irtm e nt c :o d e D e ptCc 


JB 


B-e-p-e Irnent na me-; D e ptN am e 


Departme 


DeptCode 

DeptName 

Building 


CrUp dated 






Update A. 




Drag the CrUpdated field from the 
field list to the detail section. 


Select the check box tool 
from the toolbox. 





A check box is a control 
that can be bound to fields 
of the yes/no data type. 
When the box is checked, 
True is stored in the 
table; when the box is 
unchecked, False is 
stored. 
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Tutorial exercises 


FIGURE 13.8: Add a Setvaiue command to set the value of the update status field when the 

update is compete. 


Pick the SetValue command 
from the list or simply type it in. 


mcrUp date Credits : Macro 


S etWa rninqs 
OpenQuery 
S etWa rninqs 
SetValue 


_■«.! I I 

7 


! Expression Builder 


Fo rm s! [frm D e p artm e nts]! [CrU p d ate d] 




- - 




Item 

Expression 





[CrUpdated] 


- Yes „ 




■ 


/ “| a| - 


©Tables 
©Queries 
© Foirm 
|_ fj) Lo ad e d Fo rm s 


The Item argument is the thing you 
want the SetValue action to set the mi Forms 
value of. You can use the builder or sorts 
simply type in CrUpdate. 


frmDepartments 


[<Form> 


ctions 


J 




<Field List> 


DeptCode Label 
DeptCode 
DeptName Le 
Dep 

Building Labe 
Building 
cmdUpdateC 
CrUpdated 
Labels 


□ 


OK 


Cancel 


t! Undo 


= 

> 

< 

<> | 

And 

Or 

Not 

Like | 

l| ( 

) 


Paste 


Help 


| B u i I d i n g 


CrUpdated 


I DeptCode 

I Pl ci ntM :=i rvi □ 


The Expression argument is the 
value you want the SetValue 
action to set the value of the Item 
to. Type in Yes (no quotation 
marks are required since Yes is 
recognized as a constant in this 
context). 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


• Select View> Conditions to display the condi¬ 
tions column in the macro editor as shown in 
Figure 13.9. 

FIGURE 13.9: Display the macro editors 

condition column 


Microsoft Access 


File Edit |^§gw | Insert Run Tools Window Help 
g | Macro Names 


^ Conditions 


'oolbars... 



UICUICUIIO . ITIULrIU 



Condition 

Action 


■ 

/ 

jSAtW'arnings 



/ 

OpenQuery 



/ 

SetWarnings 



/ 

SetValue 



K 




“conditions” button on the tool bar. 


13.3.5.1 The simplest conditional macro 

If there is an expression in the condition column of a 
macro, the action in that row will execute if the condi¬ 
tion is true. If the condition is not true, the action will 
be skipped. 

• Fill in the condition column as shown in 
Figure 13.10. Precede the actions you want to 
execute if the check box is checked with [Crup- 
dated] . Precede the actions you do not want to 
execute With Not [CrUpdated] . 



Since CrUpdated is a Boolean (yes/no) vari¬ 
able, you do not need to write [CrUpdated] 

= True or [CrUpdated] = False. The 

true and false parts are implied. However, if a 
non-Boolean data type is used in the expres¬ 
sion, a comparison operator must be included 

(e.g., [DeptCode] = "COMM", [Cred¬ 
its] <3, etc.) 
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FIGURE 13.10: Create a conditional macro to control which actions execute. 


The expression Not [CrUpdated] 
is true if the CrUpdated check box is 
not checked. Use this expression in 
front of the actions you want to execute 
in this situation. 


The expression [CrUpdated] is 
true if the CrUpdated check box is 
checked. In this situation, you should 
indicate to the user that the update is 
not being performed. 


The MsgBox action displays a 
standard Windows message box. You 
can set the message and other message 
box features in the arguments section. 


Action Arguments 


Courses for this department have alre 



mcrUp date Credits 

: Macro 


L ln|x| 

r 

Condition 

Action 

Comment A | 

i 

Not 

'CrUpdated' 

SetWarnings 

J 


Not 

CrUpdated' 

OpenQuery 



Not 

'CrUpdated' 

SetWarnings 



Not 

'CrUpdated' 

SetValue 


m 

[CrUpdated' 

MsgBox 

■wM? 1 

4 - 


d 


Enter the text 
of the 

message to 
display in the 
message 
box. Press 
FI for help 
on this 
argument. 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


• Switch to the form and test the macro by pressing 
the button. If the Crupdated check box is 
checked, you should get a message similar to 
that shown in Figure 13.11 . 

FIGURE 13.11: The action query is not executed 
and the message box appears instead. 


B1 Departments 


D e p s.rtrn e nt co d e [| GO M M 

Department name jCommerce and Business Administn 
Building lANGU" ^ Credits updated? 



13.3.5.2 Refining the conditions 

The macro shown in Figure 13.10 can be improved 
by using an ellipsis (...) instead of repeating the 
same condition in line after line. In this section, you 
will simplify your conditional macro slightly. 

Move the message box action and condition to the 
top of the list of actions by dragging its record selec¬ 
tor (grey box on the left). 

• Insert a new row immediately following the mes¬ 
sage and add a stopMacro action, as shown in 
Figure 13.12. 

The macro in Figure 13.12 executes as follows: If 
crupdate is true (i.e., the box is checked), the 
MsgBox action executes. Since the next line has an 
ellipsis in the condition column, the condition contin¬ 
ues to apply. However, that action on the ellipsis line 
is stopMacro, and thus the macro ends without 
executing the next four lines. 
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Tutorial exercises 


FIGURE 13.12: Rearrange the macro actions and 

insert a new row. 


Click the record selector and drag the 
message box action to the top of the list. 


w: mcrUp dale Credits : Macro 


MsgBox 



Action 


SetWarnings 


OpenQuery 

S etWa rnings 

SetValue 


Add an ellipsis 
(...) and a 
StopMacro 
action. 



#■: mcrUpdateCredits : Macro 



ConpHfion 

ik Action 


rCrUpf^tedl 

MsgBox 

pq 


StopMacro 



SetWarnings 



OpenQuery 



SetWarnings 



SetValue 


Right-click where you would like 
to insert a new row and select 
Insert Row from the popup menu. 


If the Crupdate box is not checked, the first two 
lines are ignored (i.e., the lines with the false condi¬ 
tion and the ellipsis) and the update proceeds. 

13.3.5.3 Creating a group of named macros 

It is possible to store a number of related macros 
together in one macro “module”. These group mac¬ 
ros have two advantages: 

1 . Modular macros can be created — instead of 
having a large macro with many conditions and 
branches, you can create a small macro that call 
other small macros. 

2. Similar macros can be grouped together — for 

example, you could keep all you Departments- 
related macros or search-related macros in a 
macro group. 

In this section, we will focus on the first advantage. 

• Select View> Macro Names to display the macro 
name column. 
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13. Event-Driven Programming Using Macros 


Tutorial exercises 


• Perform the steps in Figure 13.13 to modularize 
your macro. 

• Change the macro referred to in the On Click 
property of the cmdUpdateCredits button from 

mcrUpdateCredits to 
mcrUpdateCredits.CheckStatus. 

• Test the operation of the button. 

13.3.6 Creating switchboards 

One of the simplest (but most useful) triggers is an 
openForm command attached to a button on a form 
consisting exclusively of buttons. 

This type of “switchboard” (as shown in 

Figure 13.14) can provide the user with a means of 

navigating the application. 

• Create an unbound form as shown in 
Figure 13.15. 


• Remove the scroll bars, navigation buttons, and 
record selectors from the form using the form’s 
property sheet. 

• Save the form as swbMain. 

There are two ways to add button-based triggers to a 

form: 

1. Turn the button wizard off, create the button, and 
attach an macro containing the appropriate 
action (or actions). 

2. Turn the button wizard on and use the wizard to 
select from a list of common actions (the wizard 
writes a VBA procedure for you). 



Since the wizard can only attach one action to 
a button (such as opening a form or running 
an action query) it is less flexible than a 
macro. However, once you are more comfort¬ 
able with VBA, there is nothing to stop you 
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FIGURE 13.13: Use named macros to modularize the macro. 
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FIGURE 13.14: A switchboard interface to the application. 


The command buttons are placed on an 
unbound form. Note the absence of scroll bars, 
record selectors, or navigation buttons. 


Although it is not shown here, switchboards can 
call other switchboards, allowing you to add a 
hierarchical structure to your application. 


II Main Switchboard 



H5iH 

Unlvtrslty Information Suitam 


View Departments! 


Rollback Credit Update 



View Courses 


View Sections 


Gratuitous clip art can be used to 
clutter your forms and reduce the 
application’s overall performance. 


Shortcut keys are include on each 
button to allow the user to navigate 
the application with keystrokes. 
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New Form 



Choose the table or query where 
the object's data comes from: 


OK 


Desiqn View 


Form Wizard 


AutoForm: 

AutoForm: 

AutoForm: 


Columnar 

Tabular 

Datasheet 


Chart Wizard 
PivotTable Wizard 


Cancel 



FIGURE 13.15: Create an unbound form as the 
switchboard background. 


Select Design View (no wizard) and 
leave the “record source” box empty. 


The result is a blank form on which 
you can build your switchboard. 


I;s Forml : Form 
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from editing the VBA modules created by the 
wizard to add additional functionality. 

13.3.6.1 Using a macro and manually-created 
buttons 

• Ensure the wizard is turned off and use the but¬ 
ton tool to create a button. 

• Modify the properties of the button as shown in 
Figure 13.16. 

• Create a macro called 
mcrSwitchboard. OpenDept and use the 
OpenForm command to open the form f rmDe- 
partment s. 

• Attach the macro to the On Click event of the 

cmdDepartments button. 

• Test the button. 

13.3.6.2 Using the button wizard 

• Turn the button wizard back on and create a new 
button. 


• Follow the directions provided by the wizard to 
set the action for the button (i.e., open the f rm- 
Courses form) as shown in Figure 13.17. 

• Change the button’s font and resize it as 
required. 



You can standardize the size of your form 
objects by selecting more than one and using 
Format > Size > to Tallest and to Widest com¬ 
mands. Similarly, you can select more than 
one object and use the “multiple selection” 
property sheet to set the properties all at 
once. 


13.3.7 Using an autoexec macro 

If you use the name autoexec to save a macro (in 
lieu of the normal mcr<name> convention), Access 
will execute the macro actions when the database is 
opened. Consequently, auto-execute macros are 
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FIGURE 13.16: Create a button and modify its appearance. 


Use the button tool to create a button 
(ensure the wizard activated). 


M swbMain : Form 
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Give the button a meaningful name 
(e.g., cmdDepartment s) and caption 


- n x 



-I 


il] swbMain : Form 


4 - Detail 


-i 


View 

Departments 




E“ 


Scroll down the property sheet and change 
the value of the button’s Font Size property. 
Resize the button by dragging its handles. 


(including a shortcut key.). 


TEEl l 


4 ^ 


Command Button: cmdDeptartments 


All 




Format | Data Event I Other 

Name. 

Caption. 

Picture. 

Picture Type 
T ran s parent 

Default. 

Cancel. 


Status Bar Text.. 


Display When 

Enabled. 

Tab Stop. 

Tab Index.... 
Left........... 



El 


cmdDeptartments 

Vi e w 3, D e p artm e nts 

(none) 

Embedded 

No 

No 

No 

No 


Yes 

Always 

Yes 

Yes 

0 

0.698 cm 
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FIGURE 13.17: Use the command button wizard to create a button for the switchboard. 


Command Button Wizard 


Sample: 



Select Form Operations > Open Form as 
the action type associated with the button. 

What action\ag you want to happen when the button is pressed? 


Different acti 


available for each category. 


Categories: 


Record Navi Ration 
Record Ope nations 


Form Operations 


Report Operations 
Application 


Command Button Wizard 


Sample: 


HI 






Provide a caption 
for the button. 


Command Button Wizard 


frm Courses Mar 
frmCoursesUB 
frm Departments 
sfrmSections 
swbMain 


Sample: 


View 
Courses 


Select the correct form 
from the list. 


Do you wanyext or a picture on the button? 

If you choose\Text you can type the text to 
Picture, you dii click Browse to find a picti 



View & Courses 


MS Access Form 
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often used to display a switchboard when the user 
starts the application. 

Another typical auto-execute operation is to hide the 
database window. By doing this, you unclutter the 
screen and reduce the risk of a user accidentally 
making a change to the application (by deleting a 
database object, etc.). 


from the menu system. Consequently, you need to 
know something about the menu structure of Access 
before you create your macro. 

A in version 8.0, the DoMenuitem action has 
. been replaced by the slightly more intuitive 
RunCommand action. See on-line help for 
more information on RunCommand. 



To unhide the database window, select Win¬ 
dow > Unhide from the main menu or press 
the database window icon ( ®l) on the toolbar. 


The problem with hiding the database window using 
a macro is that there is no HideDatabaseWindow 
command in the Access macro language. As such, 
you have to rely on the rather convoluted DoMenu¬ 
item action. 

As its name suggests, the DoMenuitem action per¬ 
forms an operation just as if it had been selected 


• Create an auto-execute macro 

• Add the DoMenuitem and OpenForm actions to 
hide the database window and open the main 
switchboard, as shown in Figure 13.18. 

• Close the database and reopen it after a short 
delay to test the macro. 



In version 7.0 and above, you do not need to 
use an autoexec macro to hide the database 
window and open a form. Instead, you can 
right-click on the database window, select 


I ome 1 Previous] 24 of 26 | Next ► ] 













































































13. Event-Driven Programming Using Macros 


Discussion 


FIGURE 13.18: Create an auto-execute macro. 


1 # £ 

lutoexec : Macro 


Action 


g 

DoMenuItem 


OpenForm 









Menu Bar Database 

Menu Name Window 

Command^^ Hide 

Subcafnrnand 


For the DoMenuItem action, select the 
Window > Hide commands from the 
Database menu (i.e., the menu that is active 
when the database window is being used). 


Startup, and fill in the properties for the appli¬ 
cation. 


13.4 Discussion 

13.4.1 Event-driven programming versus 
conventional programming 

The primary advantages of event-driven program¬ 
ming are the following: 

1 . Flexibility — since the flow of the application is 
controlled by events rather than a sequential pro¬ 
gram, the user does not have to conform to the 
programmer’s understanding of how tasks should 
be executed. 

2. Robustness — Event-driven applications tend to 
be more robust since they are less sensitive to 
the order in which users perform activities. In 
conventional programming, the programmer has 
to anticipate virtually every sequence of activities 
the user might perform and define responses to 
these sequences. 
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The primary disadvantage of event-driven programs 
is that it is often difficult to find the source of errors 
when they do occur. This problem arises from the 
object-oriented nature of event-driven applications— 
since events are associated with a particular object 
you may have to examine a large number of objects 
before you discover the misbehaving procedure. 
This is especially true when events cascade (i.e., an 
event for one object triggers an event for a different 
object, and so on). 

13.5 Application to the assignment 

• Add “update status” check boxes to you transac¬ 
tion processing forms (i.e., Orders and Ship¬ 
ments) 

• Create a conditional macro for your shipments 
form to prevent a particular shipment from being 
added to inventory more than once. 


Application to the assignment 

Create a main switchboard for you application. It 
should provide links to all the database objects 
your user is expected to have access to (i.e., your 
forms). 
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Access Tutorial 14: Data Access Objects 


14.1 Introduction: What is the DAO 
hierarchy? 

The core of Microsoft Access and an important part 
of Visual Basic (the stand-alone application develop¬ 
ment environment) is the Microsoft Jet database 
engine. The relational DBMS functionality of Access 
comes from the Jet engine; Access itself merely pro¬ 
vides a convenient interface to the database engine. 

Because the application environment and the data¬ 
base engine are implemented as separate compo¬ 
nents, it is possible to upgrade or improve Jet 
without altering the interface aspects of Access, and 
vice-versa. 

Microsoft takes this component-based approach fur¬ 
ther in that the interface to the Jet engine consists of 
a hierarchy of components (or “objects”) called Data 
Access Objects (DAO). The advantage of DAO is 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 25-Aug-1997 


that its modularity supports easier development and 
maintenance of applications. 

The disadvantage is that is you have to understand a 
large part of the hierarchy before you can write your 
first line of useful code. This makes using VBA diffi¬ 
cult for beginners (even for those with considerable 
experience writing programs in BASIC or other 
3GLs). 

14.1.1 DAO basics 

Although you probably do not know it, you already 
have some familiarity with the DAO hierarchy. For 
example, you know that a Database object (such as 
univO_vx.mdb) contains other objects such as 
tables (TableDef objects) and queries (QueryDef 
objects). Moving down the hierarchy, you know that 
TableDef objects contain Field objects. 


* Third-generation programming languages. 
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Unfortunately, the DAO hierarchy is somewhat more 
complex than this. However, at this level, it is suffi¬ 
cient to recognize three things about DAO: 

1. Each object that you create is an instance of a 
class of similar objects (e.g., univO_vx is a par¬ 
ticular instance of the class of Database objects). 

2. Each object may contain one or more Collec¬ 
tions of objects. Collections simply keep all 
objects of a similar type or function under one 
umbrella. For example, Field objects such as 
Deptcode and crsNum are accessible through a 
Collection called Fields). 

3. Objects have properties and methods (see 
below). 

14.1.2 Properties and methods 

You should already be familiar with the concept of 
object properties from the tutorial on form design 
(Tutorial 6). The idea is much the same in DAO: 


every object has a number of properties that can be 
either observed (read-only properties) or set (read/ 
write properties). For example, each TableDef (table 
definition) object has a read-only property called 
DateCreated and a read/write property called Name. 
To access an object’s properties in VBA, you nor¬ 
mally use the <object name>.<property 
name> syntax, e.g., 

Employees.DateCreated. 



To avoid confusion between a property called 
DateCreated and a field (defined by you) 
called DateCreated, Access version 7.0 
and above require that you use a bang (!) 
instead of a period to indicate a field name or 
some other object created by you as a devel¬ 
oper. For example: 

Employees!DateCreated.Value 

identifies the Value property of the DateCre- 
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ated field (assuming one exists) in the 
Employees table. 

Methods are actions or behaviors that can be 
applied to objects of a particular class. In a sense, 
they are like predefined functions that only work in 
the context of one type of object. For example, all 
Field objects have a method called Fieidsize that 
returns the size of the field. To invoke a object’s 
methods, you use the 

<object name>.<method> [parameter^ 

. . ., parameter n ] syntax, e.g.,: 

DeptCode.Fieldsize. 



A reasonable question at this point might be: 
Isn’t Fieidsize a property of a field, not a 
method? The answer to this is that the imple¬ 
mentation of DAO is somewhat inconsistent in 
this respect. The best policy is to look at the 


object summaries in the on-line help if you are 
unsure. 

A more obvious example of a method is the cre- 
ateFieid method of TableDef objects, e.g.: 

Employees.CreateField("Phone ", 
dbText, 25) 

This creates a field called Phone, of type dbText (a 
constant used to represent text), with a length of 25 
characters. 

14.1.3 Engines, workspaces, etc. 

A confusing aspect of the DAO hierarchy is that you 
cannot simply refer to objects and their properties as 
done in the examples above. As Figure 14.1 illus¬ 
trates, you must include the entire path through the 
hierarchy in order to avoid any ambiguity between, 
say, the Deptcode field in the courses TableDef 
Object and the DeptCode field in the qryCourses 
QueryDef object. 

- I ^Tlomel M Previous | 3 of 22 | Next ► | 


14. Data Access Objects 


Introduction: What is the DAO hierarchy? 


FIGURE 14.1: Navigating the DAO hierarchy. 


© 


To access a particular field, you 
have to understand the structure 
of the DAO hierarchy. 


TableDefs 



other tables. 


DeptCode 


Indexes 



DBEngine 


Workspaces 



Databases 


IT 


By creating a database object at 
the start of your VBA 
programs, you bypass the top 
part of the hierarchy. 


other classes... 


QueryDefs 


Recordsets 


other classes.. 


qryCourses other queries... 


Fields 


DeptCode 


Indexes 


Legend 


TableDefs 


Courses 


object or collection 


instance 
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14. Data Access Objects 


Learning objectives 


Working down through the hierarchy is especially 
confusing since the first two levels (DBEngine and 
Workspaces) are essentially abstractions that have 
no physical manifestations in the Access environ¬ 
ment. The easiest way around this is to create a 
Database object that refers to the currently open 
database (e.g., univO_vx.mdb) and start from the 
database level when working down the hierarchy. 
Section 14.3.1 illustrates this process for version 2.0. 

14.2 Learning objectives 

□ What is the DAO hierarchy? 

□ What are objects? What are properties and 
methods? 

□ How do I create a reference to the current 
database object? Why is this important? 

□ What is a recordset object? 

□ How do I search a recordset? 


14.3 Tutorial exercises 

14.3.1 Setting up a database object 

In this section you will write VBA code that creates a 
pointer to the currently open database. 

• Create a new module called basDAOTest (see 
Section 12.3.3 for information on creating a new 
module). 

• Create a new subroutine called PrintRecords. 

• Define the subroutine as follows: 

Dim dbCurr As DATABASE 
Set dbCurr = 

DBEngine.Workspaces(0).Databases(0) 
Debug.Print dbCurr.Name 

• Run the procedure, as shown in Figure 14.2. 

Let us examine these three statements one by one. 

1. Dim dbCurr As DATABASE 

This statement declares the variable dbCurr as 
an object of type Database. For complex objects 
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FIGURE 14.2: Create a pointer to the current database. 


C basDAOTest: Module 

Object: 

(General) 

w 


Sub PrintRecordsf) 


lb 

© 


Declare and set the pointer 
(dbCurr) to the current 
database. 


Add a line to print the name 
of the database. 


Although you can use the 
Print statement by itself 
in the debug window, you 
must invoke the Print 
method of the Debug object 
from a module—hence the 
Debug.Print syntax. 




Dim dbCurr As DATABASE 

Set dbCurr = DBEngine.Uorkspaces(Q).Databases(Q) 
Debjg.Print dbCurr.Name 


Sub 


Debug Window 


O 

|<Ready> 


PrintRecords 
E:\uniu0_u7.mdb 


Run the procedure to 
ensure it works. 


© 


Version 7.0 and above support a less 
cumbersome way referring to the current 
database—the CurrentDb function: 
Set dbCurr = CurrentDb 
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(in contrast to simple data types like integer, 
string, etc.) Access does not allocate memory 
space for a whole database object. Instead, it 
allocates space for a pointer to a database 
object. Once the pointer is created, you must set 
it to point to an object of the declared type (the 
object may exist already or you may have to cre¬ 
ate it). 

2. Set dbCurr = DBEngine.Work¬ 
spaces (0).Databases(0) 

(Note: this should be typed on one line). In this 
statement, the variable dbCurr (a pointer to a 
Database object) is set to point to the first Data¬ 
base in the first Workspace of the only Database 
Engine. Since the numbering of objects within a 
collection starts at zero, Databases (0) indi¬ 
cates the first Database object. Note that the first 
Database object in the Databases collection is 
always the currently open one. 


A Do not worry if you are not completely sure 
. what is going on at this point. As long as you 
understand that you can type the above two 
lines to create a pointer to your database, 
then you are in good shape. 

3. Debug.Print dbCurr.Name 

This statement prints the name of the object to 
which dbCurr refers. 

14.3.2 Creating a Recordset object 

As its name implies, a TableDef object does not con¬ 
tain any data; instead, it merely defines the structure 
of a table. When you view a table in design mode, 
you are seeing the elements of the TableDef object. 
When you view a table in datasheet mode, in con¬ 
trast, you are seeing the contents of Recordset 
object associated with the table. 
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To access the data in a table using VBA, you have to 
invoke the openRecordset method of the Data¬ 
base object. Since most of the processing you do in 
VBA involves data access, familiarity with Recordset 
objects is essential. In this section, you will create a 
Recordset object based on the courses table. 

• Delete the Debug. Print dbCurr.Name line 
from your program. 

• Add the following: 

Dim rsCourses As Recordset 
Set rsCourses = 

dbCurr.OpenRecordset("Courses") 

The first line declares a pointer (rsCourses) to a 
Recordset object. The second line does two things: 

1. Invokes the OpenRecordset method of dbCurr 
to create a Recordset object based on the table 
named "Courses", (i.e., the name of the table is 
a parameter for the OpenRecordset method). 


Tutorial exercises 

2. Sets rsCourses to point to the newly created 
recordset. 

Note that this Set statement is different than the pre¬ 
vious one since the OpenRecordset method 
results in a new object being created (dbCurr points 
to an existing database—the one you opened when 
you started Access). 

14.3.3 Using a Recordset object 

In this section, you will use some of the properties 
and methods of a Recordset object to print its con¬ 
tents. 

• Add the following to PrintRecords: 

Do Until rsCourses.EOF 

Debug.Print rsCourses!DeptCode & " " 

Sc rsCourses ! CrsNum 

rsCourses.MoveNext 
Loop 

• This code is explained in Figure 14.3. 
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FIGURE 14.3: Create a program to loop through the records in a Recordset object. 


basDAOTest: Module 


Object: (General) 


T | Proc: 


Pi 


Sub PrintRecordsf) 

Dim dbCurr As DATABASE 
Set dbCurr = DBEngine.Uorkspaces(O) 
Dim rsCourses As Recordset 
Set rsCourses = dbCurr.OpenRecopd^e 

Do Until rsCourses.EOF 

Debug.Print rsCoursesfDeptCode & 
rsCourses.MoueNext 
Loop 


Datab 



EOF is a property of the recordset. 
It is true if the record counter has 
reached the “end of file” (EOF) 
marker and false otherwise. 


The exclamation mark (!) indicates 
that Dept Code is a user-defined 
field (rather than a method or 
property) of the recordset object. 


" " & rsCoursesfCrsNum 


End Sub 




Debug Window 


© 


<Ready> 


Since the Value property is the default property 
of a field, you do not have to use the 

<recordset> ! <field> .Value syntax. 


PrintRecords 
COMM 230 
COMM 291 
COMM 351 
MATH 407 
MATH 303 
CRUR 436 


The MoveNext method moves the 
record counter to the next record in 
the recordset. 
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14.3.4 Using the FindFirst method 

In this section, you will use the FindFirst method 
of Recordset objects to lookup a specific value in a 
table. 

• Create a new function called MyLookUp () using 
the following declaration: 

Function MyLookUp(strField As 
String, strTable As String, 
strWhere As String) As String 

An example of how you would use this function is to 
return the Title of a course from the Courses 
table with a particular Deptcode and crsNum. In 
other words, MyLookUp () is essentially an SQL 
statement without the select, from and where 
clauses. 

The parameters of the function are used to specify 
the name of the table (a string), the name of the field 
(a string) from which you want the value, and a 


Tutorial exercises 

where condition (a string) that ensures that only one 
record is found. 

For example, to get the Title of COMM 351 from 
the courses table, you would provide MyLookUp () 
with the following parameters: 

1 . “Title" — a string containing the name of the 
field from which we want to return a value; 

2. "Course" — a string containing the name of the 
source table; and, 

3. "DeptCode = 'COMM' AND CrsNum = 

'335' " — a string that contains the entire 
WHERE clause for the search. 

A Note that both single and double quotation 
. marks must be used to signify a string within a 
string. The use of quotation marks in this 
manner is consistent with standard practice in 
English. For example, the sentence: 

"He shouted, 'Wait for me.'" illus- 
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trates the use of single quotes within double 
quotes. 

• Define the MyLookUp () function as follows: 

Dim dbCurr As DATABASE 
Set dbCurr = CurrentDb 

A If you are using version 2.0, you cannot use 
. the CurrentDb method to return a pointer to 
the current database. You must use long form 

(i.e., Set dbCurr = DBEngine...) 


Dim rsRecords As Recordset 

Set rsRecords = 
dbCurr.OpenRecordset(strTable, 
dbOpenDynaset) 

A In version 2.0, the name of some of the pre- 
defined constants are different. As such, you 
must use db_open_dynaset rather than 
dbOpenDynaset to specify the type Of 
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Else 

MyLookUp = "" 

End I f 

• Execute the function with the following statement 
(see Figure 14.4): 

? MyLookUp("Title", "Courses", 

"DeptCode = f COMM f AND CrsNum = 

! 351 1 ") 

As it turns out, what you have implemented exists 
already in Access in the form of a predefined func¬ 
tion called DLookup (). 

• Execute the DLookup ( ) function by calling it in 
the same manner in which you called 

MyLookUp (). 

14.3.5 The DLookup () function 

The DLookup () function is the “tool of last resort” in 
Access. Although you normally use queries and 
recordsets to provide you with the information you 


Recordset object to be opened (the Find- 
First method only works with “dynaset” type 
recordsets, hence the need to include the 
additional parameter in this segment of code). 


rsRecords.FindFirst strWhere 

K VBA uses a rather unique convention to 
. determine whether to enclose the arguments 
of a function, subroutine, or method in paren¬ 
theses: if the procedure returns a value, 
enclose the parameters in parentheses; oth¬ 
erwise, use no parentheses. For example, in 
the line above, strWhere is a parameter of 
the FindFirst method (which does not 
return a value). 

If Not rsRecords.NoMatch() Then 
MyLookUp = 

rsRecords.Fields(strField).Value 
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need in your application, it is occasionally necessary 
to perform a stand-alone query—that is, to use the 
DLookup () function to retrieve a value from a table 
or query. 

When using DLookup () for the first few times, the 
syntax of the function calls may seem intimidating. 
But all you have to remember is the meaning of a 
handful of constructs that you have already used. 
These constructs are summarized below: 

• Functions — DLookup () is a function that 
returns a value. It can be used in the exact same 
manner as other functions, e.g., 

x = DLookup (...) is similar to 
x = cos(2*pi) . 

• Round brackets ( ) — In Access, round brackets 
have their usual meaning when grouping 
together operations, e.g., 3* (5+1). Round 
brackets are also used to enclose the arguments 
of function calls, e.g., x = cos(2*pi). 
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FIGURE 14.4: MyLookUp (): A function to find a value in a table. 


v basDAOTest: Module 

Object: 

(General) 

Free: MyLookUp 


Function MyLookUpfstrField As String, strTable As String, strlilhere As String) As String 


Dim dbCurr As DATABASE 
Set dbCurr = CurrentDb 

Dim rsRecords As Recordset 
Set rsRecords = dbCurr.OpenRecordset 


The NoMatch () method returns True if the 
FindFirst method finds no matching records, 
and False otherwise. 


rTable, dbOpenDynaset) 


rsRecords.FindFirst strUher^ 

If Not rsRecords.NoHatchf) Then 

MyLookUp = rsRecords.Fields(strField).Ualue 
Else 

MyLookUp = 

End If 


Since strField contains the name of a valid 
Field object (Tit le) in the Fields collection, 
this notation returns the value of Title. 


Debug Window 


End Function 


m _ 

|<Ready> 





? MyLookUpf"Title","Courses" , 
Financial Accounting 


DeptCode = 'COMM' AND CrsNum = , 351'")T 


14. Data Access Objects 

• Square brackets [ ] — Square brackets are not 
a universally defined programming construct like 
round brackets. As such, square brackets have a 
particular meaning in Access/VBA and this 
meaning is specific to Microsoft products. Simply 
put, square brackets are used to signify the name 
of a field, table, or other object in the DAO hierar¬ 
chy—they have no other meaning. Square brack¬ 
ets are mandatory when the object names 
contain spaces, but optional otherwise. For 
example, [Forms] ! [frmCourses] ! [Dept¬ 
Code] is identical to Forms ! frm- 

Courses!DeptCode. 

• Quotation marks “ ” — Double quotation marks 
are used to distinguish literal strings from names 
of variables, fields, etc. For example, 

x = "comm" means that the variable x is equal 
to the string of characters COMM. In contrast, 
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Tutorial exercises 

x = comm means that the variable x is equal to 
the value of the variable comm. 

• Single quotation marks ‘ ’ — Single quotation 
marks have only one purpose: to replace normal 
quotation marks when two sets of quotation 
marks are nested. For example, the expression 
x = "[ProductiD] = '12 3'" means that the 
variable x is equal to the string ProductiD = 

“123”. In other words, when the expression is 
evaluated, the single quotes are replaced with 
double quotes. If you attempt to nest two sets of 
double quotation marks (e.g., x = " [Produc¬ 
tiD] = "123"") the meaning is ambiguous 
and Access returns an error. 

• The Ampersand & — The ampersand is the con¬ 
catenation operator in Access/VBA and is unique 
to Microsoft products. The concatenation opera¬ 
tor joins two strings of text together into one 
string of text. For example, 
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x = "one" & "_two" means that the variable 
x is equal to the string one_two. 

If you understand these constructs at this point, then 
understanding the DLookUp () function is just a mat¬ 
ter of putting the pieces together one by one. 

14.3.5.1 Using DLookUp ( ) in queries 

The DLookUp ( ) function is extremely useful for per¬ 
forming lookups when no relationship exists between 
the tables of interest. In this section, you are going to 
use the DLookUp () function to lookup the course 
name associated with each section in the Sections 
table. Although this can be done much easier using a 
join query, this exercise illustrates the use of vari¬ 
ables in function calls. 

• Create a new query called qryLookupTest 
based on the Sections table. 

• Project the DeptCode, CrsNum, and Section 

fields. 


• Create a calculated field called Title using the 
following expression (see Figure 14.5): 

Title: DLookUp ("Title", "Courses", 
"DeptCode = [DeptCode] & AND 

CrsNum = '" & [CrsNum] & "'") 

14.3.5.2 Understanding the WHERE clause 

The first two parameters of the DLookUp () are 
straightforward: they give the name of the field and 
the table containing the information of interest. How¬ 
ever, the third argument (i.e., the where clause) is 
more complex and requires closer examination. 

At its core, this where clause is similar to the one 
you created in Section 5.3.2 in that it contains two 
criteria. However, there are two important differ¬ 
ences: 

1. Since it is a DLookUp () parameter, the entire 
clause must be enclosed within quotation marks. 
This means single and double quotes-within- 
quotes must be used. 
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FIGURE 14.5: Create a query that uses DLookUp (). 


Create a query based on the Sections 
table only (do not include Courses). 


lb 


Si Zoom 


Use the DLookUp () function to get the 
correct course title for each section. 


a 


aip qryLookUpTest 


| Sections 

■Jc 

DeptCode 


CrsNum 


Section 


Session 


| CatalogNum ▼ 11 


JJ 


Title: DLookUp( M Titie 11 /'Courses 11 /'DeptCode = 111 & [DeptCode] & 111 
AND CrsNum = 111 & [CrsNum] &.) 


OK 


Cancel 


as® qryLookUpTest: Select Query 





Field: 

DeptCode 

( 

Table: 

Sections 

C 

■_ 

Sort: 



Show: 

)/ 




Department code 

Course number 

Section 

f Title 


COMM 

351 

002 

Financial Accounting 


COMM 

351 

003 

Financial Accounting 


COMM 

439 

001 

Ad va ncedTopicsin Info rm ati o n S y ste mi s 


CRVVR 

202 

001 

Creative Forms 


CRVV'R 

202 

901 

Creative Forms 


CRVV'R 

202 

902 

Creative Forms 


CRVVR 

496 

001 

Poetry Tutorial 
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14. Data Access Objects 


Discussion 


2. It contains variable (as opposed to literal) criteria. 
For example, [Deptcode] is used instead of 
"comm". This makes the value returned by the 
function call dependent on the current value of 
the DeptCode field. 

In order to get a better feel for syntax of the function 
call, do the following exercises (see Figure 14.6): 

Switch to the debug window and define two string 
variables (see Section 12.3.1 for more information 
on using the debug window): 

strDeptCode = "COMM" 
strCrsNum = "351" 

These two variables will take the place the field val¬ 
ues while you are in the debug window. 

• Write the where clause you require without the 
variables first. This provides you with a template 
for inserting the variables. 

• Assign the where clause to a string variable 
called strwhere (this makes it easier to test). 


• Use strWhere in a DLookUp () call. 

14.4 Discussion 

14.4.1 VBA versus SQL 

The PrintRecords procedure you created in 
Section 14.3.3 is interesting since it does essentially 
the same thing as a select query: it displays a set of 
records. 

You could extend the functionality of the Print- 
Records subroutine by adding an argument and an 
if-then condition. For example: 

Sub PrintRecords(strDeptCode as 
String) 

Do Until rsCourses.EOF 

If rsCourses!DeptCode = strDeptCode 
Then 

Debug.Print rsCourses!DeptCode & " " 
& rsCourses ! CrsNum 
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FIGURE 14.6: Examine the syntax of the WHERE clause. 


Create string variables that refer to valid 
values of DeptCode and CrsNum. 


Debug Window 


□ 

|<Ready> 


Write the WHERE clause using literal 
*** criteria first to get a sense of what is 
required. 


strDeptCode 
strCrsNum = 


j^strUhere 

4 ? 


= "COMM" 

"351" 

"DeptCode = 'COMM' AND CrsNum = 
"DeptCode = '" & strDeptCode & 


/ 


\ 


strlilhere 
DeptCode = 



/ 




Use the variables in the WHERE 
clause and assign the expression to a 
string variable called strwhere. 


CrsNum = 


■ ii 


& strCrsNum & 


COMM' AND CrsNum = '351' 


? DLookUp("Title", "Courses", strlilhere) 
Financial Accounting 




To save typing, use strwhere as the 
third parameter of the DLookUp () 
call. 


© 


When replacing a literal string with a variable, you 
have to stop the quotation marks, insert the variable 
(with ampersands on either side) and restart the 
quotation marks. This procedure is evident when the 
literal and variable version are compared to each other. 
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End I f 

rsCourses.MoveNext 
Loop 

rsCourses.Close 
End Sub 

This subroutine takes a value for Deptcode as an 
argument and only prints the courses in that particu¬ 
lar department. It is equivalent to the following SQL 
command: 

SELECT DeptCode, CourseNum FROM 
Courses WHERE DeptCode = 
strDeptCode 

14.4.2 Procedural versus Declarative 

The difference between extracting records with a 
query language and extracting records with a pro¬ 
gramming language is that the former approach is 

declarative while the latter is procedural. 


SQL and QBE are declarative languages because 
you (as a programmer) need only tell the computer 
what you want done, not howto do it. In contrast, 
VBA is a procedural language since you must tell the 
computer exactly how to extract the records of inter¬ 
est. 

Although procedural languages are, in general, more 
flexible than their declarative counterparts, they rely 
a great deal on knowledge of the underlying struc¬ 
ture of the data. As a result, procedural languages 
tend to be inappropriate for end-user development 
(hence the ubiquity of declarative languages such as 
SQL in business environments). 
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14.5 Application to the assignment 

14.5.1 Using a separate table to store 
system parameters 

When you calculated the tax for the order in 
Section 9.5, you “hard-coded” the tax rate into the 
form. If the tax rate changes, you have to go through 
all the forms that contain a tax calculation, find the 
hard-coded value, and change it. Obviously, a better 
approach is to store the tax rate information in a 
table and use the value from the table in all form- 
based calculations. 

Strictly speaking, the tax rate for each product is a 
property of the product and should be stored in the 
Products table. However, in the wholesaling envi¬ 
ronment used for the assignment, the assumption is 
made that all products are taxed at the same rate. 


Application to the assignment 


As a result, it is possible to cheat a little bit and cre¬ 
ate a stand-alone table (e.g., Systemvariabies) 
that contains a single record: 


VariableName 

Value 

GST 

0.07 


Of course, other system-wide variables could be 
contained in this table, but one is enough for our pur¬ 
poses. The important thing about the SystemVari- 
abies table is that it has absolutely no relationship 
with any other table. As such, you must use a 
DLookUp () to access this information. 

• Create a table that contains information about the 
tax rate. 

• Replace the hard-coded tax rate information in 
your application with references to the value in 
the table (i.e., use a DLookUp () in your tax cal¬ 
culations). Although the SystemVariables 
table only contains one record at this point, you 
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Application to the assignment 


should use an appropriate where clause to 
ensure that the value for GST is returned (if no 
where clause is provided, DLookup () returns 
the first value in the table). 

A The use of a table such as Systemvari- 
. abies contradicts the principles of relational 
database design (we are creating an attribute 
without an entity). However, trade-offs 
between theoretical elegance and practicality 
are common in any development project. 

14.5.2 Determining outstanding 
backorders 

An good example in your assignment of a situation 
requiring use of the DLookup () is determining the 
backordered quantity of a particular item for a partic¬ 
ular customer. You need this quantity in order to cal¬ 
culate the number of each item to ship. 


The reason you must use a DLookup () to get this 
information is that there is no relationship between 

the OrderDetails and BackOrders tables. 

A Any relationship that you manage to create 
. between OrderDetails and BackOrders 

will be nonsensical and result in a non-updat- 
able recordset. 

• In the query underlying your OrderDetails 
subform, create a calculated field called QtyOn- 
BackOrder to determine the number of items on 
backorder for each item added to the order. This 
calculated field will use the DLookup () function. 

There are two differences between this DLookup () 
and the one you did in Section 14.3.5.1 

1. Both of the variables used in the function (e.g., 
CustiD and ProductiD) are not in the query. 
As such, you will have to use a join to bring the 
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missing information into the query. 

2. ProductiD is a text field and the criteria of text 
fields must be enclosed in quotation marks, e.g.: 

ProductiD = "123" 

However, CustiD is a numeric field and the crite¬ 
ria for numeric fields is not enclosed in quotations 
marks, e.g.: 

CustiD = 4 . 

A Not every combination of CustiD and Pro- 
. duct id will have an outstanding backorder. 
When a matching records is not found, the 
DLookup () function returns a special value: 
Null. The important thing to remember is 
that Null plus or minus anything equals 
Null. This has implications for your “quantity 
to ship” calculation. 

• Create a second calculated field in your query to 
convert any Nulls in the first calculated field to 


zero. To do this, use the iif () and isNuii () 
functions, e.g.: 

QtyOnBackOrderNoNull: 
iif(IsNull([QtyOnBackOrder]),0,[Qty 
OnBackOrder]) 

• Use this “clean” version in your calculations and 
on your form. 



It is possible to combine these two calculated 
fields into a one-step calculation, e.g.: 

iif (IsNull (DLookUp (...) ) , 0, 

DLookUp (...) ) . 

The problem with this approach is that the 
DLookup () function is called twice: once to 
test the conditional part of the immediate if 
statement and a second time to provide the 
“false” part of the statement. If the Back¬ 
orders table is very large, this can result in 
an unacceptable delay when displaying data 
in the form. 
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Access Tutorial 15: Advanced Triggers 


15.1 Introduction: Pulling it all 
together 

In this tutorial, you will bring together several of the 
skills you have learned in previous tutorials to imple¬ 
ment some sophisticated triggers. 

15.2 Learning objectives 

□ How do I run VBA code using a macro? 

□ How do I use the value in one field to 
automatically suggest a value for a different 
field? 

□ How do I change the table or query a form is 
bound to once the form is already created? 

□ What is the After Update event? How is it 
used? 

□ How do I provide a search capability for my 
forms? 

© Michael Brydon (brydon@unixg.ubc.ca) 

Last update: 25-Aug-1997 


□ How do I create an unbound combo box? 

□ Can I implement the search capability using 
Visual Basic? 

15.3 Tutorial exercises 

15.3.1 Using a macro to run VBA code 

There a some things that cannot be done using the 
Access macro language. If the feature you wish to 
implement is critical to your application, then you 
must implement it using VBA. However, since it is 
possible to call a VBA function from within a macro, 
you do not have to abandon the macro language 
completely. 

In this section, you are going to execute the Param- 
eterTest subroutine you created in Section 12.3.6 
from within a macro. Since the RunCode action of 
the Access macro language can only be used to exe- 
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cute functions (not subroutines) you must do one of 
two things before you create the macro: 

1. Convert ParameterTest to a function — you do 
this simply by changing the Sub at the start of the 
procedure to Function. 

2. Create a new function that executes Parame¬ 
terTest and call the function from the macro. 


’this function calls the 
ParameterTest subroutine 

ParameterTest intStart, intStop 

ParameterTestWrapper = True 
’return a value 

End Function 

• Call the function, as shown in Figure 15.1. 


15.3.1.1 Creating a wrapper 

Since the second alternative is slightly more interest¬ 
ing, it is the one we will use. 

• Open your basTesting module from 
Tutorial 12. 

• Create a new function called ParameterTest¬ 
Wrapper defined as follows: 

Function 

ParameterTestWrapper(intStart As 
Integer, intStop As Integer) As 
Integer 



Note that the return value of the function is 
declared as an integer, but the actual assign¬ 
ment Statement is ParameterTestWrap¬ 
per = True. This is because in Access/ 
VBA, the constants True and False are 
defined as integers (-1 and 0 respectively). 


15.3.1.2 Using the RunCode action 

• Leave the module open (you may have to resize 
and/or move the debug window) and create a 
new macro called mcrRunCodeTest. 


I^H ome ]E Previous] 2 of 33 | Next ► ] 
















15. Advanced Triggers 


Tutorial exercises 


FIGURE 15.1 : Create a function that calls the ParameterTest subroutine. 


© 


v basTesting : Module 

Object: 

(General) 

Proc: 

ParameterTestWrapper 



Function ParameterTestWrapperfintStart As Integer, intStop As Integer) 
'this function calls the ParameterTest subroutine 
ParameterTest intStart, intStop 
ParameterTestWn^pper = True 'return a ualue 


End Function 


Create a function to call 
the ParameterTest 
subroutine. 


Debug Window 


<Ready> 


Since ParameterTest 

does not return a value, its 
arguments are not in 
brackets. 


ParameterTestlilrapper(10,15) 
Loop number 
Loop number 
Loop number 
Loop number 
Loop number 
Loop number 
All done 
True 



Use the Print statement to 
invoke the function (do not forget 
the parameters). 



The return value of 
ParameterTestWrapper() 
is True, so this is printed when 
the function ends. 
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• Add the RunCode action and use the expression 
builder to select the correct function to execute, 
as shown in Figure 15.2. 



The expression builder includes two parame¬ 
ter place holders (<<intstart>> and 
<<intstop>>) in the function name. These 
are to remind you that you must pass two 
parameters to the ParameterTestWrap¬ 
per () function. If you leave the place holders 
where they are, the macro will fail because 
Access has not idea what <<intstart>> 
and <<intstop» refer to. 


• Replace the parameter place holders with two 
numeric parameters (e.g. 3 and 6). Note that in 
general, the parameters could be field names or 
any other references to Access objects contain¬ 
ing (in this case) integers. 


• Select Run > Start to execute the macro as 
shown in Figure 15.3. 

15.3.2 Using activity information to 

determine the number of credits 

In this section, you will create triggers attached to the 
After Update event of bound controls. 

15.3.2.1 Scenario 

Assume that each type of course activity is generally 
associated with a specific number of credits, as 
shown below: 


Activity 

Credits 

lecture 

3.0 

lab 

3.0 

tutorial 

1.0 

seminar 

6.0 
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FIGURE 15.2: Use the expression builder to select the function to execute. 


Add a RunCode 

action to the macro. 


#: mcrRunCodeTest: Macr 


Action 


lb 


Use the expression builder to drill 
down to the user-defined functions in 
your database file. 



+ 

- 

/ 



! Expression Builder 




PararneterTestW'rapper («intStart*. «iritStop A 


a 


OK 


Cancel 


- / * | & | = > < <> 


Functions 
-Fll Built-In Functions 


univO v? 


M Undo 


And Or Not Like ( ) 


Paste 


Help 


Pararn ete rTe stW'rap p e r 


IbasDAOT est 

ISBOEF 

b as Utilities 

Note the <<intStart>> and 


\ 


© 


<<intStop>> parameter place 
holders. These must be replaced 
with expressions that Access 
understands. 


P a ra m ete rTestWra p p e r(i ntSta rt, i ntSto p) 
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FIGURE 15.3: Execute the RunCode macro. 



mcrRunCodeTest: 1 

■vfacro 



Action 


■ 

RunCode 









Function Name 

Pararn ete rTe stW'rap p e r (3, 6) 

H Debug Window 


|<Ready> 

Loop number: 3 

a 

Replace the 
parameter place 
holders. 

— 

Loop number: 4 

Loop number: 5 

Loop number: 6 

All done 


lb 


Select Run > Start (or press the ! icon in 
the tool bar) to execute the macro. 


Assume as well that the number of credits for a par¬ 
ticular type of course is not cast in stone. As such, 
the numbers given above are merely “default” val¬ 
ues. 

You want to use the default credit values when you 
create a new course or modify an existing course. 
However, the user may override this default if neces¬ 
sary for a particular course. The basic requirement is 
illustrated in Figure 15.4. 

15.3.2.2 Designing the trigger 

Based on the foregoing, the answer to the “what” 
question is the following: 

1. Look up the default number of credits associated 
with the course activity showing in the form’s 

Activity field. 

2. Copy this number into the Courses . Credits 
field. 


I ome 1 Previous] 6 of 33 | Next ► ] 
































































































15. Advanced Triggers 


Tutorial exercises 


FIGURE 15.4: Inserting a default value into a new record. 
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There are several possible answers to the “when” 
question (although some are better than others). For 
example: 

1. When the user enters the Credits field (the On 
Enter event for Credits) — The problem with 
this choice is that the user could modify the 
course’s activity without moving the focus to the 
Activity field. In such a case, the trigger would 
not execute. 

2. When the user changes the Activity field (the 
After Update event for Activity) — This choice 
guarantees that whenever the value of Activ¬ 
ity is changed, the default value will be copied 
into the Credits field. As such, it is a better 
choice. 

15.3.2.3 Preliminary activities 

• Modify the Activities table to include a single¬ 
precision numeric field called credits. Add the 
values shown in the table in Section 15.3.2.1 . 


• Ensure that you have a courses form (e.g., f rm- 
courses) and that the form has a combo box for 
the Activity field. You may wish to order the 
fields such that Activity precedes credits in 
the tab order (as shown in Figure 15.4). 



If your move fields around, remember to 
adjust the tab order accordingly (recall 
Section 8.3.4). 


15.3.2.4 Looking up the default value 

As you discovered in Section 14.3.5, Access has a 
DLookup () function that allows you to go to the 
Activities table and find the value of credits 
for a particular value of Activity. A different 
approach is to join the Activities table with the 
Courses table in a query so that the default value of 
credits is always available in the form. This is the 
approach we will use here. 
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• Ensure you have a relationship (in the main rela¬ 
tionship window) between Courses .Activity 
and Activities .Activity. 

• Create a new query called qryCoursesAnd- 
Credits based on the Courses and Activi¬ 
ties tables (see Figure 15.5). 



Notice that you have two credits fields: 
courses. Credits (the actual number of 
credits for the course) and Activi¬ 
ties . credits (the “default” or “suggested” 
number of credits based on the value of 
Activity). Access uses the <table 
name>. <f ieid name> notation whenever a 
query contains more than one field with the 
same name. 


Since you already have forms based on the 
courses table that expect a field called credits 
(rather than one called courses. credits), it is a 


FIGURE 15.5: Use a join to make the default 

value available. 


is 0 qry Courses And Credits : Select Query 



Field: 

Courses* 

Credits 


Table: ' 

Courses 

Activities 


c 

. r 




o 

Sh 

Drite 

si 1 c 

qryCoursesAndCredits : Select Query 



Courses.Credits 

Activities.Credits 

Departr 


► 

2 

3 

MUSC 



3 

3 

COMM 



4 

3 

COMM 



3 

3 

COMM 



3 

3 

MATH 
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good idea to rename the Activities .credits 
field in the query. You do this by creating a calculated 
field. 

• Rename Activities . Credits to Default- 
Credits as shown in Figure 15.6. Note that this 
eliminates the need for the <tabie 
name>.<field name> notation. 

15.3.2.5 Changing the Record Source of the 
form 

Rather than create a new form based on the qry- 
CoursesAndCredits query, you can modify the 
Record Source property of the existing f rmcourses 
form so it is bound to the query rather than the 
Courses table. 

• Bring up the property sheet for the f rmcourses 
form and change the Record Source property to 

qryCoursesAndCredits as shown in 
Figure 15.7. 


FIGURE 15.6: Rename one of the credits fields. 



Field 

Table 

Sort 

Show 

Criteria 

or 


Courses * 


Courses 




DefaultCredits: Credits 


Activities 


ai 1 qryCoursesAndCredits : Select Query 



Credits 

DefaultCredits 

Deps 

► 

/ 2 

3 

MUSC 

/ 

/ 3 

3 

COMM 

/ 

4 

3 

COMM 


3 

3 

COMM 


Rename Credits form the Activities 
table to DefaultCredits. 
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FIGURE 15.7: Change the Record Source 
property of an existing form. 


BE frmCourses : !-orm 


Bring up the form’s property list and 
change its Record Source property. 


7 ' I 1 8 


* Detail 


Depa tment 


Ccmm 


5 number 


Title: 


'Activity 


Credit 


Form 


Data^l E 


qryCourse. 


DeptCode 


CrsNum 

Title 

Credits 

Activity 

DefaultCredits 


a F 



Format | Dat 
Record Source 

Filter. 

Order By... 

Allow Filters 
Caption .... 

Default View. 

Views Allowed. 

Allow Edits. 

Allow Deletions .... 
Allow Additions.... 

Data Entry. 

Recordset Type ... 

Record Locks. 

Scroll Bars. 


Event Other- 


All 




Yes 

Courses 

Single Form 

Both 

Yes 

Yes 

Yes 

No 

Dynaset 

No Locks 

Both 


= © 


The field list now contains all 
the fields in the new query. 


The advantage of using a join query in this manner is 
that DefaultCredits is now available for use 
within the form and within any macros or VBA mod¬ 
ules that run when the form is open. 

15.3.2.6 Creating the Setvaiue macro 

The Setvaiue macro you require here is extremely 
simple once you have DefaultCredits available 
within the scope of the form. 

• Create the mercourses. Setcredits macro 
as shown in Figure 15.8. 

15.3.2.7 Attaching a procedure to the After 
Update event 

The On Click event of a button is fairly simple to 
understand: the event occurs when the button is 
clicked. The events associated with non-button 
objects operate in exactly the same way. For exam¬ 
ple, the After Update event for controls (text box, 
combo box, check box, etc.) occurs when the value 
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FIGURE 15.8: Create the Set value macro. 


Create a macro group called mcrCourses 
and a named macro called SetCredits. 


#■: mcrCourses : Macro 


if Macro Name 


Action 


SetCredits 


SetValue 


Item 

Expression 


[Credits] 


[DefaultCredits 



lb 


You can use the builder to set the arguments 
or simply type in the names of the fields. 


• Attach the mcrCourses . SetCredits macro to 
the After Update event of the Activity field. 

• Verify that the trigger works properly. 

15.3.3 Use an unbound combo box to 
automate search 

As mentioned in Tutorial 8, a combo box has no 
intrinsic search capability. However, the idea of scan¬ 
ning a short list of key values, selecting a value, and 
having all the information associated with that record 
pop on to the screen is so basic that in Access ver¬ 
sion 7.0 and above, this capability is included in the 
combo box wizard. In this tutorial, we will look at a 
couple of different means of creating a combo boxes 
for search from scratch. 


of the control is changed by the user. As a result, the 
After Update event is often used to trigger data verifi¬ 
cation procedures and “auto-fill” procedures like the 
one you are creating here. 


15.3.3.1 Manual search in Access 

To see how Access searches for records, do the fol¬ 
lowing: 

• Open your frmDepartments form. 
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• Move to the field on which you want to search 
(e.g., DeptCode); 

• Select Edit > Find (or press Control-F ); 

• Fill out the search dialog box as shown in 
Figure 15.9. 

In the dialog box, you specify what to search for 
(usually a key value) and specify how Access should 
conduct its search. When you press Find First, 
Access finds the first record that matches your 
search value and makes it the current record (note 
that if you are searching on a key field, the first 
matching record is also the only matching record). 

15.3.3.2 Preliminaries 

To make this more interesting, assume that the f rm- 
Departments form is for viewing editing existing 
departmental information (rather than adding new 
departments). To enforce this limitation, do the fol¬ 
lowing: 

• Set the form’s Allow Additions property to No. 


• Set the Enabled property of DeptCode to No (the 
user should never be able to change the key val¬ 
ues of existing records). 

15.3.3.3 Creating the unbound combo box 

The key thing to remember about the combo box 
used to specify the search criterion is that it has 
nothing to do with the other fields or the underlying 
table. As such, it should be unbound. 

• Create an unbound combo box in the form 
header, as shown in Figure 15.10. 

• Change the Name property of the combo box to 

cboDeptCode. 

• The resulting combo box should resemble that 
shown in Figure 15.11 . 

A When you create an unbound combo box, 

. Access gives it a default name (e.g., 
ComboS). You should do is change this to 
something more descriptive (e.g., cboDept- 
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FIGURE 15.9: Search for a record using the “find” dialog box. 


HI Departments 


Move the cursor to 
the field you wish to 
search and invoke 
the search box 
using Control-F. 


Record: 


Find in field: 'Department cod' 


lb 


d 


Enter the value you wish to find 
and set the other search 
parameters as required. 


Limit the search to the current 
field (i.e., the field with the 
focus when the search box was 
opened). 


Press Find First to move to the first 
(or only) record that matches the 
search condition. 



IS Departments 


Department co Je |MUSC 
Department oaffie [Music 





|| CO MM 



|Commerce and Business Administn 

Update Credits 

(angu 

1* Credits updated? 



|MUSC 

Find First 


| y\.|| r Match Case P 

Find Next 

1 Search Fields As Formatted / 

|Whole Field zJ : g§ Search Only Currant Field / 

Close 




Update Credits 


IM U S C I jCredtts u p d ate d ? 

f I h |he| of 7 
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FIGURE 15.10: Create an unbound combo box. 
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FIGURE 15.11: An unbound combo box. 



Code). The advantage of the prefix cbo is 
that it allows you to differentiate between the 
bound field Deptcode and the unbound 
combo box. 


15.3.3.4 Automating the search procedure 
using a macro 

When we implement search functionality with a 
combo box, only two things are different from the 
manual search in Figure 15.9: 

1. the search dialog box does not show up, and 

2. the user selects the search value from the combo 
box rather than typing it in. 

The basic sequence of actions, however, remains 
the same. As a result, the answer to the “what” ques¬ 
tion is the following: 

1. Move the cursor to the Deptcode field (this 
allows the “Search Only Current Field” option to 
be used, thereby drastically cutting the search 
time). 

2. Invoke the search feature using the current value 
of cboDeptcode as the search value. 
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3. Move the cursor back to cboDeptCode or some 
other field. 

The only problem with this procedure is that the 
Deptcode text box is disabled. As a result, you must 
include an extra step at the beginning of the macro 
to set its Enabled property to Yes and another at the 
end of the macro to return it to its original state. 

• Create a new macro called mcrsearch .Find- 
Department. 

• Use the SetValue action to set the Dept¬ 
code . Enabled property to Yes. This can be 
done using the expression builder, as shown in 
Figure 15.12. 

• Use the GotoControi action to move the cursor 
to the Deptcode text box. Note that this action 
will fail if the destination control is disabled. 

• Use the FindRecord action to implement the 
search as shown in Figure 15.13. 


FIGURE 15.13: Fill in the arguments for the 

FindRecord action. 



Macro [lame 


Action 


FindDepartment 


SetValue 


GoTo Control 


FindRecord 


enable the DeptCode field 


move to the DeptCode field 
search 


Find What 
Match 

Match Case 
Search 
Search As Forma 
Only Current Fie I 
Find First 



Action Arguments 



= cb o D e ptCo d e]. Val u e 


Whole Field 


£ 


© 


Yes 


Yes 


Since Value is 
the default 
property, its use 
is optional. 


Enter the action arguments. Do not forget the 
equals sign before the name of the combo box. 
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FIGURE 15.12: Use the builder to specify the name of the property to set. 


To set the Item argument, use the 
expression builder to drill down 
to the correct form. 


© 



i Expression Builder 


\E\ 


F o rm s! [f rm D e p artm e nts] I [cb o D e ptCo d e]. E n ab I e d 


The middle pane shows all the 
objects on the form including 
labels and buttons (hence the 
need for a good naming 
convention). 


lb 


Select the unbound combo box 
(cboDeptCode) from the middle 
pane. A list of properties for the 
selected object is displayed in the 
pane on the right. 



+ 

- 

/ 


= 

> 

< 

<> 1 

And 

Or 

Not 

Like | 

1 

) 


(+] Tables 
ft Queries 
ft Forms 

-ft Loaded Forms 
Lfe| 

-ft All Forms, 
ft Re pi 
+TFunT 

ilH I 


frmDep ailment: 


Y 


<Form> 

<Field List> 

Se arch for a departrn e 


cboDeptCode 


DeptCode Label 
DeptCode 
DeptName Label 
Dej 

Building Label 
Building 

cm dUpdate Credits 


Ml 


OK 


Cancel 


jd 


Undo 


Paste 


Help 




DefauItValue 

DisplayWhen 


hnabled 111 

EvantProcRrefix 

J 

FcwBold 

Fontltalic 


-PontName 


FontSize 


FontUnderline 


FontW'eight 


ForeColor 
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Access interprets any text in the Find What 
argument as a literal string (i.e., quotation 
marks would not be required to find comm). To 
use an expression (including the contents of a 
control) in the Find What argument, you must 
precede it with an equals sign (e.g., 


= [cboDeptCode] . 


• You cannot disable a control if it has the focus. 
Therefore, include another GotoControi action 
to move the cursor to cboDeptCode before set¬ 
ting DeptCode.Enabled = No. 

• Attach the macro mcrSearch .FindDepart- 
ment to the After Update event Of the cboDept¬ 
Code combo box. 

• Test the search feature. 


15.3.4 Using Visual Basic code instead of 
a macro 


Instead of attaching a macro to the After Update 
event, you can attach a VBA procedure. The VBA 
procedure is much shorter than its macro counter¬ 
part: 

1. a copy (clone) of the recordset underlying the 
form is created, 

2. the FindFirst method of this recordset is used 
to find the record of interest. 

3. the “bookmark” property of the clone is used to 
move to the corresponding bookmark for the 
form. 


To create a VBA search procedure, do the following: 

• Change the After Update event of cboDeptCode 
to “Event Procedure”. 

• Press the builder (■■■) to create a VBA subrou¬ 
tine. 
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• Enter the two lines of code below, as shown in 
Figure 15.14. 

Me.RecordsetClone.FindFirst 
"DeptCode = & cboDeptCode & 

Me.Bookmark = 

Me.RecordsetClone.Bookmark 

This program consists of a number of interesting ele¬ 
ments: 

• The property Me refers to the current form. You 
can use the form's actual name, but Me is much 
faster to type. 

• A form’s RecordsetClone property provides a 
means of referencing a copy of the form's under¬ 
lying recordset. 

• The FindFirst method is straightforward. It 
acts, in this case, on the clone. 

• Every recordset has a bookmark property that 
uniquely identifies each record. A bookmark is 
like a “record number”, except that it is stored as 


Application to the assignment 

a non-human-readable data type and therefore is 
not of much use unless it is used in the manner 
shown here. Setting the Bookmark property of a 
record makes the record with that bookmark the 
current record. In the example above, the book¬ 
mark of the records underlying the form is set to 
equal the bookmark of the clone. Since the clone 
had its bookmark set by the search procedure, 
this is equivalent to searching the recordset 
underlying the form. 

15.4 Application to the assignment 

15.4.1 Triggers to help the user 

• Create a trigger on your order form that sets the 
actual selling price of a product to its default 
price. This allows the user to accept the default 
price or enter a new price for that particular trans¬ 
action (e.g., the item could be damaged). You will 
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FIGURE 15.14: Implement the search feature using a short VBA procedure. 


ii* Combo Box: cboDeptCode 


Format | Data | Event | Other 
Before Update. 

After Update. [Event Procedure! 

On Change. 


On Not in List. 

On Enter. 

On Exit. 

On Got Focus 
On Lost Focus 

On Click. 

On Dbl Click.. 
On 


[Event Procedure] 


mcrCourses 
m crCo u rs es.S etCre d its 
mcrSearch 
m crS e arch. Fi n d D e p art 
mcrUp date Credits 
m crU p d ate Cre d its. Ch e 


m crU p d ate Cre d its. U p c 


Form frmDepartments : Module 


Object: cboDeptCode 



Change the After Update event to 
reference an event procedure. 


Access automatically names the 
subroutine. Enter the two lines of code. 


IMH 


y~3 


Proc: 


AfterUpdate 


Priuate Sub cboDeptCode_Afterllpdatef) 

Me.RecordsetClone.FindFirst "DeptCode = 1 " & cboDeptCode £ . 

Me.Bookmark = Me.RecordsetClone.Bookmark 
End Sub 
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have to think carefully about which event to 
attach this macro to. 

• Create a trigger on your order form that calcu¬ 
lates a suggested quantity to ship and copies this 
value into the quantity to ship field. The sug¬ 
gested value must take into account the amount 
ordered by the customer, any outstanding backo¬ 
rders for that item by that customer, and the cur¬ 
rent quantity on hand (you cannot ship what you 
do not have). The user should be able to override 
this suggested value. (Hint: use the Minvaiue () 
function you created in Section 12.5.) 

• Provide you customer and products forms with 
search capability. 

15.4.2 Updating the BackOrders table 

Once a sales order is entered into the order form, it 
is a simple matter to calculate the amount of each 
product that should be backordered (you did this in 


Application to the assignment 

Section 10.4). The problem is updating the Back- 
Orders table itself because two different situations 
have to be considered: 

1. A record for the particular customer-product 
combination exists in the Backorders table - 

If a backorder record exists for a particular cus¬ 
tomer and a particular product, the quantity field 
of the record can be added-to or subtracted-from 
as backorders are created and filled. 

2. A customer-product record does not exist in 
the Backorders table - If the particular cus¬ 
tomer has never had a backorder for the product 
in question, then there is no record in the Back- 
Orders table to update. If you attempt to update 
a nonexistent record, you will get an error. 

What is required, therefore, is a means of determin¬ 
ing whether a record already exists for a particular 
customer-product combination. If a record does 
exist, then it has to be updated; if a record does not 
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exist, then one has to be created. This is simple 
enough to talk about, but more difficult to implement 
in VBA. As a result, you are being provided with a 
shortcut function called UpdateBackOrders () 
that implements this logic. 

The requirements for using the UpdateBackO¬ 
rders () function are outlined in the following sec¬ 
tions: 

15.4.2.1 Create the pqryltemsToBackOrder 

query 

If you have not already done so, create the pqry- 
itemsToBackOrder query described in 
Section 10.4. The UpdateBackOrders () proce¬ 
dure sets the parameter for the query and then cre¬ 
ates a recordset based on the results. 

A If you did not use the field names orderiD, 

. and Product id in your tables, you must use 
the calculated field syntax to rename them 


(see Section 15.3.2.4 to review renaming 
fields in queries). 

Note that if the backordered quantity is positive, 
items are backordered. If the backordered quantity is 
negative, backorders are being filled. If the backor¬ 
dered quantity is zero, no change is required and 
these records should no be included in the results of 
the query. 

15.4.2.2 Import the shortcut function 

Import the Visual Basic for Applications (VBA) mod¬ 
ule containing the code for the 
UpdateBackOrders () function. This module is 
contained in an Access database called 
B0sc_vx.mdb that you can download from the 
course home page. 

• bosc_v 2 . mdb is for those running Access ver¬ 
sion 2.0. To import the module, select File > 
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Import, choose bosc_v 2 .mdb, and select Mod¬ 
ule as the object type to import. 

• bosc_v 7 .mdb is for those running Access ver¬ 
sion 7.0 or higher. To import the module, select 
File > Get External Data > Import, choose 
bosc_v 7 .mdb, and select Module as the object 
type to import. 

15.4.2.3 Use the function in your application 

The general syntax of the function call is: 

UpdateBackOrders(OrderiD, CustomerlD) . 

The OrderiD and CustomerlD are arguments and 
they both must be of the type Long Integer. If this 
function is called properly, it will update all the backo¬ 
rdered items returned by the parameter query. 

15.4.2.4 Modifying the UpdateBackOrders() 
function 

The UpdateBackOrders () function looks for spe¬ 
cific fields in three tables: Backorders, custom- 
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ers, and Products. If any of your tables or fields 
are named differently, an error occurs. To eliminate 
these errors, you can do one of two of things: 

1. Edit the VBA code. Use the search-and-replace 
feature of the module editor to replace all 
instances of field names in the supplied proce¬ 
dures with your own field names. This is the rec¬ 
ommended approach, although you need an 
adequate understanding of how the code works 
in order to know which names to change. 

2. Change the field names in your tables (and all 
queries and forms that reference these field 
names). This approach is not recommended. 

15.4.3 Understanding the 

UpdateBackOrders () function 

The flowchart for the UpdateBackOrders () func¬ 
tion is shown in Figure 15.15. This function repeat¬ 
edly calls a subroutine, BackOrderitem, which 
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updates or adds the individual items to the Backo¬ 
rders table. The flowchart for the BackOrderitem 
subroutine is shown in Figure 15.16. 

There are easier and more efficient ways of imple¬ 
menting routines to update the Backorders table. 
Although some amount of VBA code is virtually inev¬ 
itable, a great deal of programming can be elimi¬ 
nated by using parameter queries and action 
queries. Since queries run faster than code in 
Access, the more code you replace with queries, the 
better. 

A To get full marks for the backorders aspect of 
. the assignment, you have to create a more 
elegant alternative to the shortcut supplied 
here. 


FIGURE 15.15: Flowchart for 


UpdateBackOrders() . 
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FIGURE 15.16: Flowchart for the BackOrderitem subroutine. 
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15.4.4 Annotated source code for the 
backorders shortcut module. 

In the following sections, the two procedures in the 
shortcut module are examined. In each case, the 
code for the procedure is presented followed by 
comments on specific lines of code. 

15.4.4.1 The UpdateBackOrders () function 

Function UpdateBackOrders(ByVal 
IngOrdID As Long, ByVal IngCustID As 
Long) 

Set dbCurr = CurrentDb 
Dim rsBOItems As Recordset 

dbCurr.QueryDef s!pqryItemsToBackOrder. 
Parameters!pOrderID = IngOrdID 

Set rsBOItems - 

dbCurr.QueryDef s!pqryIternsToBackOrder 
.OpenRecordset() 

If rsBOItems.RecordCount = 0 Then 


15. Advanced Triggers 

keyword simply means that a copy of the variables 
value is passed the subroutine, not the variable 
itself. As a result, variables passed by value cannot 
be changed by the sub-procedure. In contrast, if a 
variable is passed by reference (the default), its 
value can be changed by the sub-procedure. 

Set dbCurr = CurrentDb — Declaring a vari¬ 
able and setting it to be equal to something are dis¬ 
tinct activities. In this case, the variable dbCurr 
(which is declared in the declarations section) is set 
to point to a database object. Note that the database 
object is not created, it already exists. 

CurrentDb is a function supported in Access ver¬ 
sion 7.0 and higher that returns a reference to the 
current database. In Access version 2.0, this function 
does not exist and thus the current database must 
be found by starting at the top level object in the 
Access DAO hierarchy, as discussed in 
Section 14.3.1 . 


MsgBox "Back order cannot be processed: 
order contains no items" 

Exit Sub 
End I f 

Do Until rsBOItems.EOF 

Call BackOrderItem(IngCustID, 
rsBOItems!ProductID, rsBOItems!Qty) 

rsBOItems.MoveNext 
Loop 

rsBOItems.Close 
End Function 

15.4.4.2 Explanation of the 

UpdateBackOrders () function 

Function UpdateBackOrders(ByVal IngOr¬ 
dID As Long, ByVal IngCustID As Long) — 

This statement declares the function and its parame¬ 
ters. Each item in the parameter list contains three 
elements: ByVal or ByRef (optional), the variable's 
name, and the variable's type (optional). The ByVal 
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Dim rsBOItems As Recordset — In this decla¬ 
ration statement, a pointer to a Recordset object is 
declared. This recordset contains a list of all the 
items to add to the Backorders table. 

dbCurr.QueryDefs!pqryltemsToBackOrder 
. Parameters !pOrderlD = IngOrdID —This 

one is a bit tricky: the current database (dbCurr) 
contains a collection of objects called QueryDefs 
(these are what you create when you use the QBE 
query designer). Within the collection of QueryDefs, 
there is one called pqryltemsToBackOrder 
(which you created in Section 15.4.2.1). 

Within every QueryDef, there is a collection of zero 
or more Parameters. In this case, there is one called 
pOrderlD and this sets the value of the parameter 
to the value of the variable ingOrderiD (which was 
passed to the function as a parameter). 

Set rsBOItems = dbCurr.QueryDefs!pqry¬ 
ltemsToBackOrder . OpenRecordset () — Here 
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is another set statement. In this one, the variable 
rs bo items is set to point at a recordset object. 
Unlike the current database object above, however, 
this recordset does not yet exist and must be created 
by running the pqryitemsToBackOrder parame¬ 
ter query. 

openRecordset is a method that is defined for 
objects of type TableDef or QueryDef that creates an 
image of the data in the table or query. Since the 
query in question is a parameter query, and since the 
parameter query is set in the previous statement, the 
resulting recordset consists of a list of backordered 
items with an order number equal to the value of 

pOrderID. 

If rsBOItems.RecordCount = 0 Then -The 

only thing you need to know at this point about the 
RecordCount property of a recordset is that it returns 
zero if the recordset is empty. 


15. Advanced Triggers 

readability of this function. Note the way in which the 
current values of Product id and Qty from the 
rsBoitems Recordset are accessed. 

rsBOItems .MoveNext — MoveNext is a method 

defined for recordset objects. If this is forgotten, the 
eof condition will never be reached and an infinite 
loop will be created. In VBA, the Escape key is usu¬ 
ally sufficient to stop an infinite loop. 

Loop — All Do While/Do Until loops must end 
with the Loop statement. 

rsBoitems .close — When you create a new 
object (such as a Recordset using the Open- 
Recordset method), you should close it before exit¬ 
ing the procedure. Note that you do not close 
dbCurr because you did not open it. 

End Function — All functions/subroutines need 

an End Function/End Sub Statement. 


MsgBox "Back order cannot be processed: 
order contains no items"— The MsgBox 

statement pops up a standard message box with an 
Okay button in the middle. 

Exit Sub — If this line is reached, the list contains 
no items. As such, there is no need to go any further 
in this subroutine. 

End if — The syntax for if... Then... Else... state¬ 
ments requires an End if statement at the end of 
the conditional code. That is, everything between the 
if and the End if executes if the condition is true; 
otherwise, the whole block of code is ignored. 

Do Until rsBOItems . EOF— The EOF property 
of a recordset is set to true when the “end of file” is 
encountered. 

Call BackOrderItem(IngCustID, rsBOI- 
tems!ProductID, rsBOItems!Qty) — A sub¬ 
routine is used to increase the modularity and 
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15.4.4.3 The BackOrderItem () subroutine 

Sub BackOrderItem(ByVal IngCustID As 
Long, ByVal strProdID As String, ByVal 
intQty As Integer) 

Set dbCurr = CurrentDb 

Dim strSearch As String 

Dim rsBackOrders As Recordset 

Set rsBackOrders = 
dbCurr.OpenRecordset("BackOrders", 
dbOpenDynaset) 

strSearch = "CustID = " & IngCustID & " 
AND ProductID = 1 " & strProdID & 

rsBackOrders.FindFirst strSearch 
If rsBackOrders.NoMatch Then 
Dim rsCustomers As Recordset 

Set rsCustomers = 
dbCurr.OpenRecordset("Customers", 
dbOpenDynaset) 

strSearch = "CustID = " & IngCustID 
rsCustomers.FindFirst strSearch 
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If rsCustomers.NoMatch Then 

MsgBox "An invalid Customer ID number 
has been passed to BackOrderItem" 

Exit Sub 
End I f 

Dim rsProducts As Recordset 

Set rsProducts = 
dbCurr.OpenRecordset("Products", 
dbOpenDynaset) 

strSearch = "ProductID = f " & strProdID 

& " f " 

rsProducts.FindFirst strSearch 
If rsProducts.NoMatch Then 
MsgBox "An invalid Product ID number 
has been passed to BackOrderltem" 

Exit Sub 
End I f 

rsBackOrders.AddNew 
rsBackOrders!CustID = IngCustID 
rsBackOrders!ProductID = strProdID 


15. Advanced Triggers 

a table name, a query name, or an SQL statement. 
The dbOpenDynaset argument is a predefined con¬ 
stant that tells Access to open the recordset as a 
dynaset. You don't need to know much about this 
except that the format of these predefined constants 
is different between Access version 2.0 and version 
7.0 and higher. In version 2.0, constants are of the 
form: db_open_dynaset. 

strSearch = "CustID = "& IngCustID & " 
AND ProductID = '" & strProdID & " ! "— 

A string variable has been used to break the search 
process into two steps. First, the search string is 
constructed; then the string is used as the parameter 
for the FindFirst method. The only tricky part here 
is that IngCustID is a long integer and strProdID 
is a string. The difference is that the value of str¬ 
ProdID has to be enclosed in quotation marks when 
the parameter is passed to the FindFirst method. To 


rsBackOrders!Qty = intQty 
rsBackOrders.Update 
Else 

rsBackOrders.Edit 

rsBackOrders!Qty = rsBackOrders!Qty + 
intQty 

rsBackOrders.Update 
End I f 
End Sub 

15.4.4.4 Explanation Of the BackOrderltem ( ) 
subroutine 

Since many aspects of the language are covered in 
the previous subroutine, only those that are unique 
to this subroutine are explained. 

Set rsBackOrders = dbCurr.OpenRecord¬ 
set ( "BackOrder s" A dbOpenDynaset) —The 
OpenRecordset method used here is the one 
defined for a Database object. The most important 
argument is the source of the records, which can be 
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Application to the assignment 

do this, single quotes are used within the search 
string. 

rsBackOrders.FindFirst strSearch — 
FindFirst is a method defined for Recordset 
objects that finds the first record that meets the crite¬ 
ria specified in the method's argument. Its argument 
is the text string stored in strSearch. 

If rsBackOrders.NoMatch Then -The 
NoMat ch property should always be checked after 
searching a record set. Since it is a Boolean variable 
(True / False) it can be used without an comparison 
operator. 

rsBackOrders .AddNew — Before information can 
be added to a table, a new blank record must be cre¬ 
ated. The AddNew method creates a new empty 
record, makes it the active record, and enables it for 
editing. 
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15. Advanced Triggers 


Application to the assignment 


rsBackOrders!CustID = IngCustID— Note 

the syntax for changing a variable’s value. In this 
case, the null value of the new empty record is 
replaced with the value of a variable passed to the 
subroutine. 

rsBackOrders. Update — After any changes are 
made to a record, the update method must be 
invoked to “commit” the changes. The AddNew / 
Edit and update methods are like bookends 
around changes made to records. 

rsBackOrders.Edit - — The Edit method allows 
the values in a record to be changed. Note that these 
changes are not saved to the underlying table until 
the update method is used. 


I ^Home I Previous] 


33 of 33 | Next ► | 












